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

879 lines
33 KiB
C++

// Win2000Dom.cpp: implementation of the CWin2000Dom class.
/*---------------------------------------------------------------------------
File: Win2000Dom.cpp
Comments: Implementation of Win2K object enumeration. This object enumerates
members in any given container for Win2k domain. It
returns all information about the object that user requested.
(c) Copyright 1999, Mission Critical Software, Inc., All Rights Reserved
Proprietary and confidential to Mission Critical Software, Inc.
REVISION LOG ENTRY
Revision By: Sham Chauthani
Revised on 07/02/99 12:40:00
---------------------------------------------------------------------------
*/
#include "stdafx.h"
#include "EaLen.hpp"
#include <adserr.h>
#include "NT4Enum.h"
#include "Win2KDom.h"
#ifdef _DEBUG
#define new DEBUG_NEW
#undef THIS_FILE
static char THIS_FILE[] = __FILE__;
#endif
//////////////////////////////////////////////////////////////////////
// Construction/Destruction
//////////////////////////////////////////////////////////////////////
CWin2000Dom::CWin2000Dom()
{
}
CWin2000Dom::~CWin2000Dom()
{
mNameContextMap.clear();
}
//---------------------------------------------------------------------------
// GetEnumeration : Given the information this method returns all requested
// values for a given object in a VARIANT containing
// SAFEARRAY containing each of the property value that was
// requested by the caller.
//---------------------------------------------------------------------------
HRESULT CWin2000Dom::GetEnumeration(
BSTR sContainerName, //in- Container to enumerate ( LDAP sub path )
BSTR sDomainName, //in- Domain where the container resides
BSTR m_sQuery, //in- LDAP query string (for filtering)
long attrCnt, //in- Number of properties requested.
LPWSTR * sAttr, //in- Pointer to array of Property names.
ADS_SEARCHPREF_INFO prefInfo, //in- Search preference info.
BOOL bMultiVal, //in- Indicates whether to return multivalue props or not.
IEnumVARIANT **& pVarEnum //out- IEnumVARIANT object that will enumerate all returned objects.
)
{
// First get the full path to the container from the subpath that we have here.
_bstr_t sAdsPath;
_bstr_t sNamingContext;
_bstr_t sGrpDN;
_bstr_t sQuery;
_variant_t var, var2;
IADs * pAds = NULL;
int nCnt = 0;
// IADsMembers * pMbr = NULL;
// IADsGroup * pGrp = NULL;
HRESULT hr = S_OK;
HRESULT hr2;
IDirectorySearch * pSearch = NULL;
ADS_SEARCH_HANDLE hSearch = NULL;
TNodeList * pList = new TNodeList();
// int cnt = 0;
bool cont = true;
ADS_SEARCH_COLUMN col;
BSTR sClass = NULL;
if (!pList)
return HRESULT_FROM_WIN32(ERROR_NOT_ENOUGH_MEMORY);
//get the default naming context for this domain
sNamingContext = GetDefaultNamingContext(sDomainName);
if (sNamingContext.length())
{
if ( (wcsncmp(sContainerName, L"LDAP://", 7) != 0) && (wcsncmp(sContainerName, L"GC://", 5) != 0) )
{
// Partial path supplied so we will build the rest.
if ( sContainerName && *sContainerName )
{
sAdsPath = L"LDAP://";
sAdsPath += sDomainName;
sAdsPath += L"/";
sAdsPath += sContainerName;
sAdsPath += L",";
sAdsPath += sNamingContext;
}
else
{
sAdsPath = L"LDAP://";
sAdsPath += sDomainName;
sAdsPath += L"/";
sAdsPath += sNamingContext;
}
}
else //else full path so no need to build anything.
sAdsPath = sContainerName;
}//end if got default naming context
if (sAdsPath.length())
{
//connect to the object
hr = ADsGetObject(sAdsPath, IID_IADs, (void**) &pAds);
if (SUCCEEDED(hr))
{
hr = pAds->get_Class(&sClass);
}
if ( SUCCEEDED(hr) )
{
if ( sClass && wcscmp(sClass, L"group") == 0 && prefInfo.vValue.Integer != ADS_SCOPE_BASE )
{
// If we're trying to enumerate the contents of a group,
// Construct the DN for group and the LDAP path to whole directory
hr = pAds->Get(L"distinguishedName", &var2);
if ( SUCCEEDED(hr) )
{
sGrpDN = var2.bstrVal;
sAdsPath = L"LDAP://";
sAdsPath += sDomainName;
sAdsPath += L"/";
sAdsPath += sNamingContext;
// modify the query so that we have (& (memberOf=%s) (...) ) query
sQuery = L"(&(memberOf=";
sQuery += sGrpDN;
sQuery += L") ";
sQuery += m_sQuery;
sQuery += L")";
// Also the scope changes since we need to search the whole domain
prefInfo.vValue.Integer = ADS_SCOPE_SUBTREE;
hr = ADsGetObject(sAdsPath, IID_IDirectorySearch, (void**) &pSearch);
}
}//end if group object
else
{
sQuery = m_sQuery;
hr = pAds->QueryInterface(IID_IDirectorySearch, (void**)&pSearch);
}
SysFreeString(sClass);
}//end if got the object class
if ( SUCCEEDED(hr) )
{
hr = pSearch->SetSearchPreference(&prefInfo, 1);
// Set the query to be paged query so that we can get the data larger than a 1000.
ADS_SEARCHPREF_INFO prefInfo2;
prefInfo2.dwSearchPref = ADS_SEARCHPREF_PAGESIZE;
prefInfo2.vValue.dwType = ADSTYPE_INTEGER;
prefInfo2.vValue.Integer = 999;
if (SUCCEEDED(hr))
{
hr = pSearch->SetSearchPreference(&prefInfo2, 1);
}
}
if ( SUCCEEDED(hr) )
{
if ( (prefInfo.vValue.Integer == ADS_SCOPE_BASE) && (bMultiVal) &&
(_wcsicmp(L"member", sAttr[0]) == 0) || (_wcsicmp(L"memberOf", sAttr[0]) == 0) ||
(_wcsicmp(L"directReports", sAttr[0]) == 0) || (_wcsicmp(L"managedObjects", sAttr[0]) == 0))
{
DoRangeQuery(sDomainName, sQuery, sAttr, attrCnt, hSearch, pSearch, bMultiVal, pList);
}
else
{
hr = pSearch->ExecuteSearch(sQuery, sAttr, attrCnt, &hSearch);
if ( SUCCEEDED(hr) )
{
hr = pSearch->GetFirstRow(hSearch);
}
if ( hr == S_OK )
{
while( cont )
{
_variant_t * varAr = new _variant_t[attrCnt];
if (!varAr)
{
if ( pSearch )
{
pSearch->CloseSearchHandle(hSearch);
pSearch->Release();
}
if ( pAds )
pAds->Release();
delete pList;
return HRESULT_FROM_WIN32(ERROR_NOT_ENOUGH_MEMORY);
}
// int ncol = 0;
for ( long dw = 0; dw < attrCnt; dw++ )
{
hr = pSearch->GetColumn( hSearch, sAttr[dw], &col );
if ( SUCCEEDED(hr) )
{
if ( col.dwNumValues > 0 )
{
// Get the type of attribute and put the value into the variant
// and then into the Enumerator object.
if ( col.dwNumValues < 2 || !bMultiVal )
// put the last item into the enumeration.(memberOf=CN=DNSAdmins,CN=USERS,DC=devblewerg,DC=com)
AttrToVariant(col.pADsValues[col.dwNumValues - 1], varAr[dw]);
else
{
// Build a VARIANT array of all the values.
SAFEARRAY * pArray;
SAFEARRAYBOUND bd = {col.dwNumValues, 0};
_variant_t var;
_bstr_t strTemp;
_variant_t HUGEP * vArray;
pArray = SafeArrayCreate(VT_VARIANT|VT_BSTR, 1, &bd);
// Fill up the VARIANT Array
SafeArrayAccessData(pArray, (void HUGEP **) &vArray);
for ( DWORD x = 0; x < col.dwNumValues; x++ )
{
nCnt++;
AttrToVariant(col.pADsValues[x], var);
strTemp = var;
vArray[x] = _variant_t(strTemp);
}
SafeArrayUnaccessData(pArray);
varAr[dw].vt = VT_ARRAY | VT_VARIANT;
SafeArrayCopy(pArray, &varAr[dw].parray);
}
}
else
{
// Put an empty string here.
varAr[dw] = (BSTR)NULL;
hr = S_OK;
}
pSearch->FreeColumn( &col );
}
else
{
// Put an empty string here.
varAr[dw] = (BSTR)NULL;
if (hr == E_ADS_COLUMN_NOT_SET)
{
hr = S_OK;
}
}
}
TAttrNode * pNode = new TAttrNode(attrCnt, varAr);
if (!pNode)
{
delete [] varAr;
if ( pSearch )
{
pSearch->CloseSearchHandle(hSearch);
pSearch->Release();
}
if ( pAds )
pAds->Release();
delete pList;
return HRESULT_FROM_WIN32(ERROR_NOT_ENOUGH_MEMORY);
}
// Clear the array
delete [] varAr;
hr2 = pSearch->GetNextRow(hSearch);
if ( hr2 == S_ADS_NOMORE_ROWS ) {
cont = false;
}
else if (FAILED(hr2)) {
if ( pSearch )
{
pSearch->CloseSearchHandle(hSearch);
pSearch->Release();
}
if ( pAds )
pAds->Release();
delete pList;
return hr2;
}
pList->InsertBottom(pNode);
}
}
pSearch->CloseSearchHandle(hSearch);
}
}
}//end if got adspath
else
{
hr = HRESULT_FROM_WIN32(ERROR_CANT_ACCESS_DOMAIN_INFO);
}
if ( pSearch )
pSearch->Release();
if ( pAds )
pAds->Release();
if (FAILED(hr))
{
delete pList;
return hr;
}
// UpdateAccountInList(pList, sDomainName);
*pVarEnum = (IEnumVARIANT *) new CNT4Enum(pList);
if (!(*pVarEnum))
{
delete pList;
return HRESULT_FROM_WIN32(ERROR_NOT_ENOUGH_MEMORY);
}
return S_OK;
}
//--------------------------------------------------------------------
// AttrToVariant : This function stores a value from ADSValue struct
// into a variant in a appropriate type.
//--------------------------------------------------------------------
bool CWin2000Dom::AttrToVariant(
ADSVALUE pADsValues, //in- Value for a property in ADSVALUE struct
_variant_t& var //out-Variant filled with info from the in parameter.
)
{
HRESULT hr = S_OK;
// Fill in the values as per the varset.
switch (pADsValues.dwType)
{
case ADSTYPE_INVALID : break;
case ADSTYPE_DN_STRING : var = pADsValues.DNString;
break;
case ADSTYPE_CASE_EXACT_STRING : var = pADsValues.CaseExactString;
break;
case ADSTYPE_CASE_IGNORE_STRING : var = pADsValues.CaseIgnoreString;
break;
case ADSTYPE_PRINTABLE_STRING : var = pADsValues.PrintableString;
break;
case ADSTYPE_NUMERIC_STRING : var = pADsValues.NumericString;
break;
case ADSTYPE_INTEGER : var.vt = VT_I4;
var.lVal = pADsValues.Integer;
break;
case ADSTYPE_OCTET_STRING : {
var.vt = VT_ARRAY | VT_UI1;
var.parray = NULL;
byte * pData;
DWORD dwLength = pADsValues.OctetString.dwLength;
SAFEARRAY * sA;
SAFEARRAYBOUND rgBound = {dwLength, 0};
sA = ::SafeArrayCreate(VT_UI1, 1, &rgBound);
::SafeArrayAccessData( sA, (void**)&pData);
for ( DWORD i = 0; i < dwLength; i++ )
pData[i] = pADsValues.OctetString.lpValue[i];
hr = ::SafeArrayUnaccessData(sA);
hr = ::SafeArrayCopy(sA, &var.parray);
hr = ::SafeArrayDestroy(sA);
}
break;
/* case ADSTYPE_UTC_TIME : var = L"Date not supported.";
break;
case ADSTYPE_LARGE_INTEGER : var = L"Large Integer not supported.";
break;
case ADSTYPE_PROV_SPECIFIC : var = L"Provider specific strings not supported.";
break;
case ADSTYPE_OBJECT_CLASS : var = pADsValues.ClassName;
break;
case ADSTYPE_CASEIGNORE_LIST : wcscat(sKeyName,L".ERROR");
var.vt = VT_BSTR;
var.bstrVal = L"Case ignore lists are not supported.";
break;
case ADSTYPE_OCTET_LIST : wcscat(sKeyName,L".ERROR");
var.vt = VT_BSTR;
var.bstrVal = L"Octet lists are not supported.";
break;
case ADSTYPE_PATH : wcscat(sKeyName,L".ERROR");
var.vt = VT_BSTR;
var.bstrVal = L"Path type not supported.";
break;
case ADSTYPE_POSTALADDRESS : wcscat(sKeyName,L".ERROR");
var.vt = VT_BSTR;
var.bstrVal = L"Postal addresses are not supported.";
break;
case ADSTYPE_TIMESTAMP : var.vt = VT_UI4;
var.lVal = attrInfo.pADsValues[dw].UTCTime;
break;
case ADSTYPE_BACKLINK : wcscat(sKeyName,L".ERROR");
var.vt = VT_BSTR;
var.bstrVal = L"Backlink is not supported.";
break;
case ADSTYPE_TYPEDNAME : wcscat(sKeyName,L".ERROR");
var.vt = VT_BSTR;
var.bstrVal = L"Typed name not supported.";
break;
case ADSTYPE_HOLD : wcscat(sKeyName,L".ERROR");
var.vt = VT_BSTR;
var.bstrVal = L"Hold not supported.";
break;
case ADSTYPE_NETADDRESS : wcscat(sKeyName,L".ERROR");
var.vt = VT_BSTR;
var.bstrVal = L"NetAddress not supported.";
break;
case ADSTYPE_REPLICAPOINTER : wcscat(sKeyName,L".ERROR");
var.vt = VT_BSTR;
var.bstrVal = L"Replica pointer not supported.";
break;
case ADSTYPE_FAXNUMBER : wcscat(sKeyName,L".ERROR");
var.vt = VT_BSTR;
var.bstrVal = L"Faxnumber not supported.";
break;
case ADSTYPE_EMAIL : wcscat(sKeyName,L".ERROR");
var.vt = VT_BSTR;
var.bstrVal = L"Email not supported.";
break;
case ADSTYPE_NT_SECURITY_DESCRIPTOR : wcscat(sKeyName,L".ERROR");
var.vt = VT_BSTR;
var.bstrVal = L"Security Descriptor not supported.";
break;
*/
default : return false;
}
return true;
}
/*void CWin2000Dom::UpdateAccountInList(TNodeList *pList, BSTR sDomainName)
{
bool found = false;
for ( TAttrNode * pNode = (TAttrNode *)pList->Head(); pNode; pNode = (TAttrNode *)pNode->Next())
{
if ( _bstr_t(pNode->m_Val) == _bstr_t(sPDC) )
{
found = true;
break;
}
}
if ( !found )
{
TAttrNode * pNode = new TAttrNode(attrCnt, varAr);
pList->InsertBottom(pNode);
}
}
*/
HRESULT CWin2000Dom::DoRangeQuery(BSTR sDomainName, BSTR sQuery, LPWSTR * sAttr, int attrCnt, ADS_SEARCH_HANDLE hSearch, IDirectorySearch * pSearch, BOOL bMultiVal, TNodeList * pList)
{
HRESULT hr = S_OK;
HRESULT hr2;
bool cont = true;
ADS_SEARCH_COLUMN col;
int nCnt = 0;
int * pStartWith;
int * pEndWith;
WCHAR sAttrRange[LEN_Path];
int tryCols = 0;
LPWSTR * sAttrs = NULL;
_variant_t * varAr;
TAttrNode * pNode;
LPWSTR * psAttrNames;
BOOL * pDone;
BOOL bAllDone = FALSE;
int nOrigCnt = attrCnt;
int ndx;
pStartWith = new int[attrCnt];
pEndWith = new int[attrCnt];
psAttrNames = new LPWSTR[attrCnt];
sAttrs = new LPWSTR[attrCnt];
pDone = new BOOL[attrCnt];
if ((!pStartWith) || (!pEndWith) || (!psAttrNames) || (!sAttrs) || (!pDone))
{
if (pStartWith)
delete [] pStartWith;
if (pEndWith)
delete [] pEndWith;
if (psAttrNames)
delete [] psAttrNames;
if (sAttrs)
delete [] sAttrs;
if (pDone)
delete [] pDone;
return HRESULT_FROM_WIN32(ERROR_NOT_ENOUGH_MEMORY);
}
for (ndx = 0; ndx < attrCnt; ndx++)
{
pStartWith[ndx] = 0;
pEndWith[ndx] = 0;
psAttrNames[ndx] = sAttr[ndx];
sAttrs[ndx] = _wcsdup(sAttr[ndx]);
if (!sAttrs[ndx] && sAttr[ndx])
{
for (int i=0; i < ndx; i++)
{
free(sAttrs[i]);
}
delete [] pStartWith;
delete [] pEndWith;
delete [] psAttrNames;
delete [] sAttrs;
delete [] pDone;
return HRESULT_FROM_WIN32(ERROR_NOT_ENOUGH_MEMORY);
}
pDone[ndx] = FALSE;
}
// continue to retrieve field's values in MAX chunks until done
while (!bAllDone)
{
int last = 0;
for (ndx = 0; ndx < attrCnt; ndx++)
{
if (pDone[ndx] == FALSE)
{
if (IsPropMultiValued((WCHAR*)psAttrNames[ndx], (WCHAR*)sDomainName) == true)
wsprintf(sAttrRange, L"%s;range=%d-*", (WCHAR*)(psAttrNames[ndx]), pStartWith[ndx]);
else
wcscpy(sAttrRange, (WCHAR*)psAttrNames[ndx]);
free(sAttrs[ndx]);
sAttrs[last] = _wcsdup(sAttrRange);
if (!sAttrs[last])
{
hr = HRESULT_FROM_WIN32(ERROR_NOT_ENOUGH_MEMORY);
}
psAttrNames[last] = psAttrNames[ndx];
pStartWith[last] = pStartWith[ndx];
pEndWith[last] = pEndWith[ndx];
pDone[last] = pDone[ndx];
last++;
}
else
free(sAttrs[ndx]);
}
attrCnt = last;
if (hr != S_OK)
{
for (int i=0; i < attrCnt; i++)
{
free(sAttrs[i]);
}
delete [] pStartWith;
delete [] pEndWith;
delete [] psAttrNames;
delete [] sAttrs;
delete [] pDone;
return hr;
}
varAr = new _variant_t[attrCnt];
if (!varAr)
{
delete [] pStartWith;
delete [] pEndWith;
delete [] psAttrNames;
delete [] pDone;
for (ndx = 0; ndx < attrCnt; ndx++)
{
free(sAttrs[ndx]);
}
delete [] sAttrs;
return HRESULT_FROM_WIN32(ERROR_NOT_ENOUGH_MEMORY);
}
for (ndx=0; ndx<attrCnt; ndx++)
{
varAr[ndx] = (BSTR)NULL;
pDone[ndx] = TRUE;
}
hr = pSearch->ExecuteSearch(sQuery, sAttrs, attrCnt, &hSearch);
if ( SUCCEEDED(hr) )
{
hr = pSearch->GetFirstRow(hSearch);
}
if ( hr == S_OK )
{
while( cont )
{
LPWSTR pszColumn;
_bstr_t sTemp;
//since the column name could have changed (in the case that there are
//more values to enumerate than IDirectorySearch can in a single call :
//default is 1000) we need to find the column's new name
hr = pSearch->GetNextColumnName(hSearch, &pszColumn);
while (pszColumn != NULL)
{
int current = -1;
if ((SUCCEEDED(hr)) && (hr != S_ADS_NOMORE_COLUMNS))
{
//get the new column name
do
{
current++;
sTemp = psAttrNames[current];
if (wcslen(psAttrNames[current]) != (wcslen(pszColumn)))
sTemp += L";range=";
}
while ((current < attrCnt) && (wcsstr(pszColumn, (WCHAR*)sTemp) == NULL));
pDone[current] = FALSE;
if (wcsstr(pszColumn, (WCHAR*)sTemp) != NULL)
{
_bstr_t oldName = psAttrNames[current];
oldName += L";range=";
if ((wcsstr(pszColumn, oldName) != NULL) &&
(wcsstr(pszColumn, L"-*") == NULL))
{
WCHAR sName[MAX_PATH];
//now get the new range max retrieved so far
swscanf(pszColumn, L"%[^;];range=%d-%d", sName, &pStartWith[current], &pEndWith[current]);
free(sAttrs[current]);
sAttrs[current] = _wcsdup(pszColumn); //save the new column name
if (!sAttrs[current])
{
hr = HRESULT_FROM_WIN32(ERROR_NOT_ENOUGH_MEMORY);
}
}
else if ((wcsstr(pszColumn, L"-*") != NULL) || (!wcscmp(pszColumn, psAttrNames[current])))
pDone[current] = TRUE;
FreeADsMem(pszColumn);
if (hr == HRESULT_FROM_WIN32(ERROR_NOT_ENOUGH_MEMORY))
{
delete [] varAr;
delete [] pStartWith;
delete [] pEndWith;
delete [] psAttrNames;
delete [] pDone;
for (ndx = 0; ndx < attrCnt; ndx++)
{
free(sAttrs[ndx]);
}
delete [] sAttrs;
pSearch->CloseSearchHandle(hSearch);
return hr;
}
}
}
hr = pSearch->GetColumn( hSearch, sAttrs[current], &col );
if ( SUCCEEDED(hr) )
{
if ( col.dwNumValues > 0 )
{
// Build a VARIANT array of all the values.
SAFEARRAY * pArray;
SAFEARRAYBOUND bd = {col.dwNumValues, 0};
_variant_t var;
_bstr_t strTemp;
_variant_t HUGEP * vArray;
pArray = SafeArrayCreate(VT_VARIANT|VT_BSTR, 1, &bd);
// Fill up the VARIANT Array
SafeArrayAccessData(pArray, (void HUGEP **) &vArray);
for ( DWORD x = 0; x < col.dwNumValues; x++ )
{
nCnt++;
AttrToVariant(col.pADsValues[x], var);
strTemp = var;
vArray[x] = _variant_t(strTemp);
}
SafeArrayUnaccessData(pArray);
varAr[current].vt = VT_ARRAY | VT_VARIANT;
SafeArrayCopy(pArray, &varAr[current].parray);
}
pSearch->FreeColumn( &col );
}
hr = pSearch->GetNextColumnName(hSearch, &pszColumn);
}//end while more columns
hr2 = pSearch->GetNextRow(hSearch);
if ( (hr2 == S_ADS_NOMORE_ROWS) || FAILED(hr2) )
{
hr = S_OK;
cont = false;
}
}
}
else
{
bAllDone = TRUE;
}
if ( pStartWith[0] == 0 )
{
pNode = new TAttrNode((long)attrCnt, varAr);
if (!pNode)
{
delete [] varAr;
delete [] pStartWith;
delete [] pEndWith;
delete [] psAttrNames;
delete [] pDone;
for (ndx = 0; ndx < attrCnt; ndx++)
{
free(sAttrs[ndx]);
}
delete [] sAttrs;
pSearch->CloseSearchHandle(hSearch);
return HRESULT_FROM_WIN32(ERROR_NOT_ENOUGH_MEMORY);
}
}
else //else add the new values for any remaining attributes
{
for (int i=0; i<attrCnt; i++)
{
int j=-1;
bool bFound = false;
//find the original column of this attribute for the 'Add' call
while ((j < nOrigCnt) && (!bFound))
{
j++;
if (wcscmp(psAttrNames[i], sAttr[j]) == 0)
bFound = true; //original column number
}
if (bFound) //if found, add the new values for that column
pNode->Add((long)j, (long)i, varAr);
}
}
// Clear the array
delete [] varAr;
pSearch->CloseSearchHandle(hSearch);
cont = true;
bAllDone = TRUE;
for (ndx = 0; ndx < attrCnt; ndx++)
{
pStartWith[ndx] = pEndWith[ndx] + 1; //start the next query
bAllDone = ((bAllDone) && (pDone[ndx])) ? TRUE : FALSE; //see if done with all properties
}
}
delete [] pStartWith;
delete [] pEndWith;
delete [] psAttrNames;
delete [] pDone;
for (ndx = 0; ndx < attrCnt; ndx++)
{
free(sAttrs[ndx]);
}
delete [] sAttrs;
pList->InsertBottom(pNode);
return hr;
}
/*********************************************************************
* *
* Written by: Paul Thompson *
* Date: 10 NOV 2000 *
* *
* This function is responsible for checking a property's schema *
* to see if that property is multi-valued or not. *
* *
*********************************************************************/
//BEGIN IsPropMultiValued
bool CWin2000Dom::IsPropMultiValued(const WCHAR * sPropName, const WCHAR * sDomain)
{
HRESULT hr;
VARIANT_BOOL bMulti = VARIANT_FALSE;
WCHAR sAdsPath[LEN_Path];
DWORD dwArraySizeOfsAdsPath = sizeof(sAdsPath)/sizeof(sAdsPath[0]);
IADsProperty * pProp = NULL;
if ( sPropName == NULL || wcslen(sPropName) == 0 )
return false;
if (sDomain == NULL)
_com_issue_error(E_INVALIDARG);
//prepare to call the property's schema
if (wcslen(L"LDAP://") + wcslen(sDomain) + wcslen(L"/")
+ wcslen(sPropName) + wcslen(L", schema") >= dwArraySizeOfsAdsPath)
_com_issue_error(HRESULT_FROM_WIN32(ERROR_INSUFFICIENT_BUFFER));
wcscpy(sAdsPath, L"LDAP://");
wcscat(sAdsPath, sDomain);
wcscat(sAdsPath, L"/");
wcscat(sAdsPath, sPropName);
wcscat(sAdsPath, L", schema");
hr = ADsGetObject(sAdsPath, IID_IADsProperty, (void **)&pProp);
// Get the Multi-Valued flag for the property
if (SUCCEEDED(hr))
{
hr = pProp->get_MultiValued(&bMulti);
pProp->Release();
}
if (FAILED(hr))
{
_com_issue_error(hr);
}
if (bMulti == VARIANT_TRUE)
return true;
else
return false;
}
//END IsPropMultiValued
/*********************************************************************
* *
* Written by: Paul Thompson *
* Date: 13 JUNE 2001 *
* *
* This function is responsible for getting the default naming *
* context for the given domain. We store the domain\naming context *
* pairs in a map class variable so that we only have to look up the *
* naming context once per instantiations of this object. *
* *
*********************************************************************/
//BEGIN GetDefaultNamingContext
_bstr_t CWin2000Dom::GetDefaultNamingContext(_bstr_t sDomain)
{
/* local variables */
HRESULT hr;
_bstr_t sNamingContext = L"";
CNameContextMap::iterator itDNCMap;
/* function body */
if (!sDomain.length())
return sNamingContext;
//look if we have already cached the naming context for this domain
itDNCMap = mNameContextMap.find(sDomain);
//if found, get the cached naming context
if (itDNCMap != mNameContextMap.end())
{
sNamingContext = itDNCMap->second;
}
else //else, lookup the naming context from scratch and add that to the cache
{
_bstr_t sAdsPath;
_variant_t var;
IADs * pAds = NULL;
// Get the default naming context for this domain
sAdsPath = L"LDAP://";
sAdsPath += sDomain;
sAdsPath += L"/rootDSE";
hr = ADsGetObject(sAdsPath, IID_IADs, (void**) &pAds);
if ( SUCCEEDED(hr))
{
hr = pAds->Get(L"defaultNamingContext", &var);
//if got the context, add it to the cache
if ( SUCCEEDED(hr) )
{
sNamingContext = var.bstrVal;
mNameContextMap.insert(CNameContextMap::value_type(sDomain, sNamingContext));
}
}
}
return sNamingContext;
}
//END GetDefaultNamingContext