mirror of https://github.com/tongzx/nt5src
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.
629 lines
17 KiB
629 lines
17 KiB
// DSQuery.cpp : Implementation of ds routines and classes
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Microsoft Windows
|
|
// Copyright (C) Microsoft Corporation, 1992 - 1999
|
|
//
|
|
// File: querysup.cpp
|
|
//
|
|
// Contents: DS Enumeration routines and classes
|
|
//
|
|
// History: 02-Oct-96 WayneSc Created
|
|
//
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
|
|
#include "stdafx.h"
|
|
#include "resource.h"
|
|
|
|
#include "dsutil.h"
|
|
|
|
#include "dssnap.h" // NOTE: this must be befroe querysup.h
|
|
#include "querysup.h"
|
|
|
|
#include "dsdirect.h"
|
|
|
|
#include <lmaccess.h>
|
|
|
|
|
|
|
|
#ifdef _DEBUG
|
|
#define new DEBUG_NEW
|
|
#undef THIS_FILE
|
|
static char THIS_FILE[] = __FILE__;
|
|
#endif
|
|
|
|
extern const INT g_nADsPath;
|
|
extern const INT g_nName;
|
|
extern const INT g_nDisplayName;
|
|
extern const INT g_nObjectClass;
|
|
extern const INT g_nGroupType;
|
|
extern const INT g_nDescription;
|
|
extern const INT g_nUserAccountControl;
|
|
extern const INT g_nSystemFlags;
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
CDSSearch::CDSSearch()
|
|
{
|
|
m_bInitialized = FALSE;
|
|
m_pwszFilter = NULL;
|
|
m_pCache = NULL;
|
|
m_pCD = NULL;
|
|
m_pObj = NULL;
|
|
m_SearchHandle = NULL;
|
|
}
|
|
|
|
CDSSearch::CDSSearch(CDSCache *pCache, CDSComponentData *pCD)
|
|
{
|
|
m_bInitialized = FALSE;
|
|
m_pwszFilter = NULL;
|
|
m_pCache = pCache;
|
|
m_pCD = pCD;
|
|
m_pObj = NULL;
|
|
m_SearchHandle = NULL;
|
|
}
|
|
|
|
|
|
void CDSSearch::_Reset()
|
|
{
|
|
if (m_pObj != NULL)
|
|
{
|
|
if (m_SearchHandle)
|
|
{
|
|
m_pObj->CloseSearchHandle (m_SearchHandle);
|
|
m_SearchHandle = NULL;
|
|
}
|
|
m_pObj->Release();
|
|
m_pObj = NULL;
|
|
}
|
|
}
|
|
|
|
CDSSearch::~CDSSearch()
|
|
{
|
|
_Reset();
|
|
}
|
|
|
|
|
|
HRESULT CDSSearch::Init(IDirectorySearch * pObj)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
_Reset();
|
|
m_pObj = pObj;
|
|
pObj->AddRef();
|
|
m_bInitialized = TRUE;
|
|
m_scope = ADS_SCOPE_ONELEVEL;
|
|
|
|
return hr;
|
|
}
|
|
|
|
HRESULT CDSSearch::Init(LPCWSTR lpszObjectPath)
|
|
{
|
|
HRESULT hr;
|
|
|
|
_Reset();
|
|
|
|
hr = DSAdminOpenObject(lpszObjectPath,
|
|
IID_IDirectorySearch,
|
|
(void **)&m_pObj);
|
|
if (SUCCEEDED(hr)) {
|
|
m_bInitialized = TRUE;
|
|
} else {
|
|
m_bInitialized = FALSE;
|
|
m_pObj = NULL;
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
HRESULT CDSSearch::SetAttributeList (LPTSTR *pszAttribs, INT cAttrs)
|
|
{
|
|
if ( !m_pszAttribs.SetCount(cAttrs) )
|
|
return E_OUTOFMEMORY;
|
|
for (INT i = 0; i < cAttrs; i++)
|
|
{
|
|
if ( !m_pszAttribs.Set(pszAttribs[i], i) )
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT CDSSearch::SetAttributeListForContainerClass (CDSColumnSet* pColumnSet)
|
|
{
|
|
ASSERT(pColumnSet != NULL);
|
|
|
|
PWSTR *pAttributes = new PWSTR[g_nStdCols + pColumnSet->GetNumCols()]; // leave extra space
|
|
if (!pAttributes)
|
|
{
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
|
|
int nCols = 0;
|
|
for (int i=0; i < g_nStdCols; i++)
|
|
{
|
|
pAttributes[nCols++] = g_pStandardAttributes[i];
|
|
}
|
|
POSITION pos = pColumnSet->GetHeadPosition();
|
|
while (pos != NULL)
|
|
{
|
|
CDSColumn* pCol = (CDSColumn*)pColumnSet->GetNext(pos);
|
|
ASSERT(pCol != NULL);
|
|
|
|
if (!(pCol->GetColumnType() == ATTR_COLTYPE_SPECIAL || pCol->GetColumnType() == ATTR_COLTYPE_MODIFIED_TIME) ||
|
|
!pCol->IsVisible())
|
|
continue;
|
|
|
|
LPWSTR pNewAttribute = const_cast<LPWSTR>(pCol->GetColumnAttribute());
|
|
|
|
//
|
|
// JonN 2/8/99: Do not query the same attribute more than once
|
|
//
|
|
for (int j = 0; j < nCols; j++)
|
|
{
|
|
if ( pNewAttribute != NULL)
|
|
{
|
|
if ( 0 == _wcsicmp( pAttributes[j], pNewAttribute ) )
|
|
{
|
|
pNewAttribute = NULL;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (NULL != pNewAttribute)
|
|
pAttributes[nCols++] = pNewAttribute;
|
|
}
|
|
|
|
// JonN 6/29/99: remember the container class name (NULL is OK)
|
|
m_strContainerClassName = pColumnSet->GetClassName();
|
|
|
|
HRESULT hr = SetAttributeList (pAttributes, nCols);
|
|
delete[] pAttributes;
|
|
pAttributes = 0;
|
|
|
|
return hr;
|
|
}
|
|
|
|
HRESULT CDSSearch::SetSearchScope (ADS_SCOPEENUM scope)
|
|
{
|
|
if (m_bInitialized) {
|
|
m_scope = scope;
|
|
}
|
|
return S_OK;
|
|
}
|
|
|
|
|
|
const int NUM_PREFS=4;
|
|
HRESULT _SetSearchPreference(IDirectorySearch* piSearch, ADS_SCOPEENUM scope)
|
|
{
|
|
if (NULL == piSearch)
|
|
{
|
|
ASSERT(FALSE);
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
ADS_SEARCHPREF_INFO aSearchPref[NUM_PREFS];
|
|
aSearchPref[0].dwSearchPref = ADS_SEARCHPREF_CHASE_REFERRALS;
|
|
aSearchPref[0].vValue.dwType = ADSTYPE_INTEGER;
|
|
aSearchPref[0].vValue.Integer = ADS_CHASE_REFERRALS_EXTERNAL;
|
|
aSearchPref[1].dwSearchPref = ADS_SEARCHPREF_PAGESIZE;
|
|
aSearchPref[1].vValue.dwType = ADSTYPE_INTEGER;
|
|
aSearchPref[1].vValue.Integer = QUERY_PAGESIZE;
|
|
aSearchPref[2].dwSearchPref = ADS_SEARCHPREF_CACHE_RESULTS;
|
|
aSearchPref[2].vValue.dwType = ADSTYPE_BOOLEAN;
|
|
aSearchPref[2].vValue.Integer = FALSE;
|
|
aSearchPref[3].dwSearchPref = ADS_SEARCHPREF_SEARCH_SCOPE;
|
|
aSearchPref[3].vValue.dwType = ADSTYPE_INTEGER;
|
|
aSearchPref[3].vValue.Integer = scope;
|
|
|
|
return piSearch->SetSearchPreference (aSearchPref, NUM_PREFS);
|
|
}
|
|
|
|
HRESULT _FRSMemberQuery( IDirectorySearch* piAnyMember, CMapStringToString& strmap )
|
|
{
|
|
#define IFTRUERETURN(b) if (b) { return E_FAIL; }
|
|
#define IFFAILRETURN(hr) if (FAILED(hr)) { return hr; }
|
|
|
|
// get path to container
|
|
CComQIPtr<IADs, &IID_IADs> spIADsContainer( piAnyMember );
|
|
IFTRUERETURN( !spIADsContainer );
|
|
CComBSTR sbstr;
|
|
HRESULT hr = spIADsContainer->get_ADsPath( &sbstr );
|
|
IFFAILRETURN(hr);
|
|
|
|
// remove leaf element from path (get path to grandparent)
|
|
CPathCracker pathCracker;
|
|
hr = pathCracker.Set(sbstr, ADS_SETTYPE_FULL);
|
|
IFFAILRETURN(hr);
|
|
hr = pathCracker.RemoveLeafElement();
|
|
IFFAILRETURN(hr);
|
|
sbstr.Empty();
|
|
hr = pathCracker.Retrieve( ADS_FORMAT_X500, &sbstr );
|
|
IFFAILRETURN(hr);
|
|
|
|
// set up search
|
|
CComPtr<IDirectorySearch> spSearch;
|
|
hr = DSAdminOpenObject(sbstr,
|
|
IID_IDirectorySearch,
|
|
(void **)&spSearch);
|
|
IFFAILRETURN(hr);
|
|
hr = _SetSearchPreference(spSearch, ADS_SCOPE_ONELEVEL);
|
|
IFFAILRETURN(hr);
|
|
DSPROP_BSTR_BLOCK bstrblockAttribs;
|
|
bstrblockAttribs.SetCount( 2 );
|
|
IFTRUERETURN( !bstrblockAttribs.Set( L"distinguishedName", 0 ) );
|
|
IFTRUERETURN( !bstrblockAttribs.Set( L"fRSComputerReference", 1 ) );
|
|
|
|
// perform search
|
|
ADS_SEARCH_HANDLE hSearch = NULL;
|
|
hr = spSearch->ExecuteSearch (L"(objectClass=nTFRSMember)",
|
|
bstrblockAttribs,
|
|
bstrblockAttribs.QueryCount(),
|
|
&hSearch);
|
|
|
|
// build mapping
|
|
hr = spSearch->GetNextRow ( hSearch );
|
|
while (hr == S_OK) {
|
|
ADS_SEARCH_COLUMN adscol;
|
|
hr = spSearch->GetColumn( hSearch, L"distinguishedName", &adscol );
|
|
IFFAILRETURN(hr);
|
|
CString strDistinguishedName;
|
|
IFTRUERETURN( !ColumnExtractString( strDistinguishedName, NULL, &adscol ) );
|
|
spSearch->FreeColumn( &adscol );
|
|
|
|
hr = spSearch->GetColumn( hSearch, L"fRSComputerReference", &adscol );
|
|
IFFAILRETURN(hr);
|
|
CString strFRSComputerReference;
|
|
IFTRUERETURN( !ColumnExtractString( strFRSComputerReference, NULL, &adscol ) );
|
|
spSearch->FreeColumn( &adscol );
|
|
|
|
strmap.SetAt( strDistinguishedName, strFRSComputerReference );
|
|
|
|
hr = spSearch->GetNextRow( hSearch );
|
|
}
|
|
IFFAILRETURN(hr);
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT CDSSearch::DoQuery()
|
|
{
|
|
BEGIN_PROFILING_BLOCK("CDSSearch::DoQuery");
|
|
|
|
if (!m_bInitialized)
|
|
return E_ADS_BAD_PATHNAME;
|
|
|
|
HRESULT hr = _SetSearchPreference(m_pObj, m_scope);
|
|
|
|
if (SUCCEEDED(hr)) {
|
|
hr = m_pObj->ExecuteSearch (m_pwszFilter,
|
|
m_pszAttribs,
|
|
m_pszAttribs.QueryCount(),
|
|
&m_SearchHandle);
|
|
}
|
|
|
|
//
|
|
// JonN 6/29/99: If we are enumerating an nTFRSMember container, we must
|
|
// now perform an auxiliary search for the fRSComputerReference attribute
|
|
// on the nTFRSMember objects which are the parent container and
|
|
// the siblings of the container.
|
|
//
|
|
if (SUCCEEDED(hr) && !m_strContainerClassName.Compare( _T("nTFRSMember") ) )
|
|
{
|
|
_FRSMemberQuery( m_pObj, m_mapMemberToComputer );
|
|
}
|
|
|
|
END_PROFILING_BLOCK;
|
|
return hr;
|
|
}
|
|
|
|
HRESULT
|
|
CDSSearch::GetNextRow()
|
|
{
|
|
BEGIN_PROFILING_BLOCK("CDSSearch::GetNextRow");
|
|
|
|
DWORD status = ERROR_MORE_DATA;
|
|
HRESULT hr = S_OK;
|
|
HRESULT hr2 = S_OK;
|
|
WCHAR Buffer1[512], Buffer2[512];
|
|
if (!m_bInitialized) {
|
|
END_PROFILING_BLOCK;
|
|
return E_ADS_BAD_PATHNAME;
|
|
}
|
|
while (status == ERROR_MORE_DATA ) {
|
|
hr = m_pObj->GetNextRow (m_SearchHandle);
|
|
if (hr == S_ADS_NOMORE_ROWS) {
|
|
hr2 = ADsGetLastError(&status, Buffer1, 512,
|
|
Buffer2, 512);
|
|
ASSERT(SUCCEEDED(hr2));
|
|
} else {
|
|
status = 0;
|
|
}
|
|
}
|
|
END_PROFILING_BLOCK;
|
|
return hr;
|
|
}
|
|
|
|
HRESULT
|
|
CDSSearch::GetColumn(LPWSTR Attribute,
|
|
PADS_SEARCH_COLUMN pColumnData)
|
|
{
|
|
if (m_bInitialized) {
|
|
return m_pObj->GetColumn (m_SearchHandle,
|
|
Attribute,
|
|
pColumnData);
|
|
}
|
|
return E_ADS_BAD_PATHNAME;
|
|
}
|
|
|
|
HRESULT
|
|
CDSSearch::SetCookieFromData(CDSCookie* pCookie,
|
|
CDSColumnSet* pColumnSet)
|
|
{
|
|
CPathCracker pathCracker;
|
|
return SetCookieFromData(pCookie, pathCracker, pColumnSet);
|
|
}
|
|
|
|
HRESULT
|
|
CDSSearch::SetCookieFromData (CDSCookie* pCookie,
|
|
CPathCracker& specialPerformancePathCracker,
|
|
CDSColumnSet* pColumnSet)
|
|
{
|
|
|
|
if (pCookie==NULL) {
|
|
ASSERT(FALSE); // Invalid Arguments
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
BEGIN_PROFILING_BLOCK("CDSSearch::SetCookieFromData");
|
|
|
|
CString str;
|
|
HRESULT hr = S_OK;
|
|
BOOL BadArgs = FALSE;
|
|
INT GroupType = 0;
|
|
ADS_SEARCH_COLUMN ColumnData, ColumnData2;
|
|
CString szClass;
|
|
|
|
// ---------- Get Path --------------
|
|
hr = m_pObj->GetColumn(m_SearchHandle,
|
|
m_pszAttribs[g_nADsPath],
|
|
&ColumnData);
|
|
if (SUCCEEDED(hr) && ColumnExtractString( str, pCookie, &ColumnData ))
|
|
{
|
|
CString szPath;
|
|
StripADsIPath (str, szPath);
|
|
pCookie->SetPath(szPath);
|
|
} else {
|
|
str.LoadString( IDS_DISPLAYTEXT_NONE );
|
|
BadArgs = TRUE;
|
|
TRACE(_T("cannot read ADsPath, tossing cookie... (hr is %lx)\n"),
|
|
hr);
|
|
ReportErrorEx (m_pCD->GetHWnd(), IDS_INVALID_ROW, S_OK,
|
|
MB_OK | MB_ICONINFORMATION, NULL, 0);
|
|
goto badargs;
|
|
}
|
|
if (SUCCEEDED(hr)) m_pObj->FreeColumn (&ColumnData);
|
|
|
|
// ---------- Get Name --------------
|
|
hr = m_pObj->GetColumn(m_SearchHandle,
|
|
m_pszAttribs[g_nName],
|
|
&ColumnData);
|
|
if (!(SUCCEEDED(hr) && ColumnExtractString( str, pCookie, &ColumnData ))) {
|
|
CString Path;
|
|
|
|
// CPathCracker pathCracker;
|
|
Path = pCookie->GetPath();
|
|
specialPerformancePathCracker.Set((LPWSTR)(LPCWSTR)Path,
|
|
ADS_SETTYPE_DN);
|
|
specialPerformancePathCracker.SetDisplayType(ADS_DISPLAY_VALUE_ONLY);
|
|
BSTR ObjName = NULL;
|
|
specialPerformancePathCracker.GetElement( 0, &ObjName );
|
|
str = ObjName;
|
|
}
|
|
pCookie->SetName(str);
|
|
|
|
if (SUCCEEDED(hr)) m_pObj->FreeColumn (&ColumnData);
|
|
|
|
// ---------- Get Class (and Group Type) --------------
|
|
hr = m_pObj->GetColumn(m_SearchHandle,
|
|
m_pszAttribs[g_nObjectClass],
|
|
&ColumnData);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
szClass = ColumnData.pADsValues[ColumnData.dwNumValues-1].CaseIgnoreString;
|
|
}
|
|
if (szClass.IsEmpty() || FAILED(hr))
|
|
{
|
|
szClass = L"Unknown";
|
|
}
|
|
else
|
|
{
|
|
HRESULT hr2 = m_pObj->GetColumn(m_SearchHandle,
|
|
m_pszAttribs[g_nGroupType],
|
|
&ColumnData2);
|
|
if (SUCCEEDED(hr2))
|
|
{
|
|
GroupType = ColumnData2.pADsValues[ColumnData2.dwNumValues-1].Integer;
|
|
m_pObj->FreeColumn (&ColumnData2);
|
|
}
|
|
}
|
|
if (SUCCEEDED(hr)) m_pObj->FreeColumn (&ColumnData);
|
|
|
|
|
|
// ---------- Get Description --------------
|
|
hr = m_pObj->GetColumn(m_SearchHandle,
|
|
m_pszAttribs[g_nDescription],
|
|
&ColumnData);
|
|
if (SUCCEEDED(hr)) {
|
|
if (ColumnExtractString( str, pCookie, &ColumnData)) {
|
|
pCookie->SetDesc(str);
|
|
}
|
|
m_pObj->FreeColumn (&ColumnData);
|
|
}
|
|
else {
|
|
pCookie->SetDesc(L"");
|
|
}
|
|
|
|
// ---------- Get AccountControl Flag word --------------
|
|
hr = m_pObj->GetColumn(m_SearchHandle,
|
|
m_pszAttribs[g_nUserAccountControl],
|
|
&ColumnData);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
if (ColumnData.pADsValues->dwType == ADSTYPE_INTEGER)
|
|
{
|
|
if (((DWORD)ColumnData.pADsValues->Integer & UF_INTERDOMAIN_TRUST_ACCOUNT) == UF_INTERDOMAIN_TRUST_ACCOUNT)
|
|
{
|
|
BadArgs = TRUE;
|
|
}
|
|
else if ((((DWORD)ColumnData.pADsValues->Integer & UF_ACCOUNTDISABLE)) != UF_ACCOUNTDISABLE)
|
|
{
|
|
pCookie->ReSetDisabled();
|
|
}
|
|
else
|
|
{
|
|
pCookie->SetDisabled();
|
|
}
|
|
|
|
if ((((DWORD)ColumnData.pADsValues->Integer & UF_DONT_EXPIRE_PASSWD)) != UF_DONT_EXPIRE_PASSWD)
|
|
{
|
|
pCookie->ReSetNonExpiringPwd();
|
|
}
|
|
else
|
|
{
|
|
pCookie->SetNonExpiringPwd();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
pCookie->ReSetDisabled();
|
|
pCookie->ReSetNonExpiringPwd();
|
|
}
|
|
m_pObj->FreeColumn (&ColumnData);
|
|
}
|
|
|
|
// ---------- Get System Flags --------------
|
|
pCookie->SetSystemFlags(0);
|
|
hr = m_pObj->GetColumn(m_SearchHandle,
|
|
m_pszAttribs[g_nSystemFlags],
|
|
&ColumnData);
|
|
if (SUCCEEDED(hr)) {
|
|
if (ColumnData.pADsValues->dwType == ADSTYPE_INTEGER) {
|
|
pCookie->SetSystemFlags(ColumnData.pADsValues->Integer);
|
|
}
|
|
m_pObj->FreeColumn (&ColumnData);
|
|
}
|
|
|
|
// ---------- Get Class Cache and Cookie Extra Info --------------
|
|
// JonN 6/17/99: moved from BadArgs clause
|
|
if (!BadArgs)
|
|
{
|
|
CString szPath;
|
|
m_pCD->GetBasePathsInfo()->ComposeADsIPath(szPath, pCookie->GetPath());
|
|
|
|
CDSClassCacheItemBase* pItem = m_pCache->FindClassCacheItem(m_pCD, szClass, szPath);
|
|
if (pItem != NULL)
|
|
{
|
|
pCookie->SetCacheItem(pItem);
|
|
if (szClass == L"group")
|
|
{
|
|
CDSCookieInfoGroup* pExtraInfo = new CDSCookieInfoGroup;
|
|
pExtraInfo->m_GroupType = GroupType;
|
|
pCookie->SetExtraInfo(pExtraInfo);
|
|
} else if (szClass == L"nTDSConnection") {
|
|
CDSCookieInfoConnection* pExtraInfo = new CDSCookieInfoConnection;
|
|
ASSERT( NULL != pExtraInfo );
|
|
|
|
hr = m_pObj->GetColumn(m_SearchHandle,
|
|
L"fromServer",
|
|
&ColumnData);
|
|
if (SUCCEEDED(hr)) {
|
|
CString strFromServer;
|
|
if ( ColumnExtractString( strFromServer, NULL, &ColumnData) ) {
|
|
CString strFRSComputerReference;
|
|
if ( m_mapMemberToComputer.Lookup( strFromServer, strFRSComputerReference ) )
|
|
{
|
|
pExtraInfo->m_strFRSComputerReference = strFRSComputerReference;
|
|
}
|
|
}
|
|
m_pObj->FreeColumn (&ColumnData);
|
|
}
|
|
|
|
hr = m_pObj->GetColumn(m_SearchHandle,
|
|
L"options",
|
|
&ColumnData);
|
|
if (SUCCEEDED(hr) && NULL != ColumnData.pADsValues) {
|
|
pExtraInfo->m_nOptions = ColumnData.pADsValues[0].Integer;
|
|
m_pObj->FreeColumn (&ColumnData);
|
|
}
|
|
|
|
pExtraInfo->m_fFRSConnection = !m_strContainerClassName.Compare( _T("nTFRSMember") );
|
|
|
|
pCookie->SetExtraInfo(pExtraInfo);
|
|
}
|
|
} else {
|
|
BadArgs = TRUE;
|
|
}
|
|
}
|
|
|
|
|
|
hr = S_OK;
|
|
|
|
// ---------- Get optional Columns -----------
|
|
if ((pColumnSet != NULL)) {
|
|
CStringList& strlist = pCookie->GetParentClassSpecificStrings();
|
|
|
|
strlist.RemoveAll(); // remove contents, if we do an update
|
|
|
|
POSITION pos = pColumnSet->GetHeadPosition();
|
|
while (pos != NULL)
|
|
{
|
|
CDSColumn* pCol = (CDSColumn*)pColumnSet->GetNext(pos);
|
|
if (!(pCol->GetColumnType() == ATTR_COLTYPE_SPECIAL || pCol->GetColumnType() == ATTR_COLTYPE_MODIFIED_TIME) ||
|
|
!pCol->IsVisible())
|
|
continue;
|
|
str = L"";
|
|
|
|
COLUMN_EXTRACTION_FUNCTION pfn = pCol->GetExtractionFunction();
|
|
if (NULL == pfn) {
|
|
pfn = ColumnExtractString;
|
|
}
|
|
|
|
hr = m_pObj->GetColumn(m_SearchHandle,
|
|
const_cast<LPWSTR>(pCol->GetColumnAttribute()),
|
|
&ColumnData);
|
|
if (SUCCEEDED(hr)) {
|
|
if ( NULL == pfn || !(pfn)( str, pCookie, &ColumnData ) ) {
|
|
str = L" ";
|
|
}
|
|
|
|
// If the column is the modified time, then copy it into the cookie as a SYSTEMTIME so that
|
|
// we can do a comparison for sorting
|
|
if (pCol->GetColumnType() == ATTR_COLTYPE_MODIFIED_TIME)
|
|
{
|
|
pCookie->SetModifiedTime(&(ColumnData.pADsValues->UTCTime));
|
|
}
|
|
FreeColumn (&ColumnData);
|
|
}
|
|
else
|
|
{
|
|
if ( NULL == pfn || !(pfn)( str, pCookie, NULL ) ) {
|
|
str = L" ";
|
|
}
|
|
}
|
|
strlist.AddTail( str );
|
|
}
|
|
}
|
|
|
|
hr = S_OK;
|
|
|
|
badargs:
|
|
if (BadArgs) {
|
|
hr = HRESULT_FROM_WIN32(ERROR_INVALID_DATA);
|
|
}
|
|
|
|
END_PROFILING_BLOCK;
|
|
return hr;
|
|
}
|