windows-nt/Source/XPSP1/NT/ds/adsi/router/csession.cxx
2020-09-26 16:20:57 +08:00

1610 lines
45 KiB
C++

//---------------------------------------------------------------------------
//
// Microsoft Windows
// Copyright (C) Microsoft Corporation, 1992 - 1995
//
// File: csession.cxx
//
// Contents: Microsoft OleDB/OleDS Session object for ADSI
//
//
// History: 08-01-96 shanksh Created.
//
//----------------------------------------------------------------------------
#include "oleds.hxx"
#pragma hdrstop
#include "atl.h"
#include "row.hxx"
//+---------------------------------------------------------------------------
//
// Function: CSessionObject::GetDataSource
//
// Synopsis: Retrieve an interface pointer on the session object
//
// Arguments:
// riid, IID desired
// ppDSO ptr to interface
//
//
// Returns:
// S_OK Session Object Interface returned
// E_INVALIDARG ppDSO was NULL
// E_NOINTERFACE IID not supported
//
// Modifies:
//
// History: 08-28-96 ShankSh Created.
//
//----------------------------------------------------------------------------
STDMETHODIMP
CSessionObject::GetDataSource (
REFIID riid,
IUnknown ** ppDSO
)
{
//
// Asserts
//
ADsAssert(_pDSO);
if( ppDSO == NULL )
RRETURN( E_INVALIDARG );
RRETURN( _pDSO->QueryInterface(riid, (LPVOID*)ppDSO) );
}
//+---------------------------------------------------------------------------
//
// Function: CSessionObject::GetDataSource
//
// Synopsis: Retrieve an interface pointer on the session object
//
// Arguments:
// riid, IID desired
// ppDSO ptr to interface
//
//
// Returns:
// S_OK Session Object Interface returned
// E_INVALIDARG ppDSO was NULL
// E_NOINTERFACE IID not supported
//
// Modifies:
//
// History: 08-28-96 ShankSh Created.
//
//----------------------------------------------------------------------------
STDMETHODIMP
CSessionObject::OpenRowset(
IUnknown * pUnkOuter,
DBID * pTableID,
DBID * pIndexID,
REFIID riid,
ULONG cPropertySets,
DBPROPSET rgPropertySets[],
IUnknown ** ppRowset
)
{
// Don't pass any credentials (NULL)
RRETURN( OpenRowsetWithCredentials(
pUnkOuter,
pTableID,
pIndexID,
riid,
cPropertySets,
rgPropertySets,
NULL,
ppRowset) );
}
//+---------------------------------------------------------------------------
//
// Function: CSessionObject::OpenRowsetWithCredentials
//
// Synopsis: Opens a rowset. Similar to OpenRowset but takes extra argument
// CCredentials. This function is used when consumer calls
// IBindResource::Bind requesting a rowset.
//
// Returns : HRESULT
//
//----------------------------------------------------------------------------
HRESULT
CSessionObject::OpenRowsetWithCredentials (
IUnknown * pUnkOuter,
DBID * pTableID,
DBID * pIndexID,
REFIID riid,
ULONG cPropertySets,
DBPROPSET rgPropertySets[],
CCredentials * pCredentials,
IUnknown ** ppRowset
)
{
BOOL fWarning = FALSE;
CRowProvider *pRowProvider = NULL;
DWORD cAttrs = 1;
BOOL *pbMultiValue = NULL;
LPWSTR *pAttrs = NULL;
//
// Check in-params and NULL out-params in case of error
//
if( ppRowset )
*ppRowset = NULL;
if( !pTableID && !pIndexID )
RRETURN( E_INVALIDARG );
if( pIndexID )
RRETURN( DB_E_NOINDEX );
//
// Check the PropertySets
//
if( cPropertySets > 0 && !rgPropertySets )
RRETURN ( E_INVALIDARG );
for(ULONG i=0; i<cPropertySets; i++) {
if( rgPropertySets[i].cProperties && !rgPropertySets[i].rgProperties )
RRETURN ( E_INVALIDARG );
}
//
// pwszName field represents the ADsPath of the Directory we have to open;
// Make sure pwszName is meaningful
//
if( !pTableID || pTableID->eKind != DBKIND_NAME ||
!pTableID->uName.pwszName || !(*(pTableID->uName.pwszName)) )
RRETURN( DB_E_NOTABLE );
if( pUnkOuter )//&& !InlineIsEqualGUID(riid, IID_IUnknown) )
RRETURN( DB_E_NOAGGREGATION );
if( riid == IID_NULL )
RRETURN( E_NOINTERFACE );
//
// By default, we use credentials stored in member variable _Credentials
// for binding. Buf if caller passed any credentials through pCredentials,
// these take precedence. This also means that we need to store these
// credentials in the CRowProvider object for future use -
// e.g. GetRowFromHRow will use these credentials.
//
CCredentials * pCreds = &_Credentials;
if( pCredentials )
pCreds = pCredentials;
//
// If integrated security is being used, impersonate the caller
//
BOOL fImpersonating;
fImpersonating = FALSE;
if(_pDSO->IsIntegratedSecurity())
{
HANDLE ThreadToken = _pDSO->GetThreadToken();
ASSERT(ThreadToken != NULL);
if (ThreadToken)
{
if (!ImpersonateLoggedOnUser(ThreadToken))
RRETURN(E_FAIL);
fImpersonating = TRUE;
}
else
RRETURN(E_FAIL);
}
HRESULT hr = GetDSInterface(
pTableID->uName.pwszName,
*pCreds,
IID_IDirectorySearch,
(void **)&_pDSSearch);
if (fImpersonating)
{
RevertToSelf();
fImpersonating = FALSE;
}
if( FAILED(hr) )
RRETURN( hr );
//
// Get ColumnsInfo based on the list of attributes that we want to be
// returned. GetDefaultColumnInfo cleansup memory on failure.
//
ULONG cColumns = 0;
DBCOLUMNINFO * prgInfo = NULL;
WCHAR * pStringBuffer = NULL;
hr = GetDefaultColumnInfo(&cColumns, &prgInfo, &pStringBuffer);
if( FAILED(hr) )
RRETURN( hr );
// Store the properties (which must be in the rowset property group) in
// the property object. OpenRowset is different from methods like
// ICOmmand::SetProperties and ISessionProperties::SetProperties in that
// it returns DB_E_ERROSOCCURRED if any property which is REQUIRED could
// not be set and DB_S_ERROROCCURRED if any property that is OPTIONAL
// could not be set. ICommand::SetProperties returns DB_E_ERROSOCCURRED
// if all properties could not be set and DB_S_ERROSOCCURRED if some
// property could not be set i.e, DBPROPOPTIONS (REQUIRED or OPTIONAL) is
// ignored.
// Use PROPSET_COMMAND as the bitmask below since the properties that are
// going to be set are in the rowset property group. These properties that
// are stored in the property object cannot be retrieved by the client
// since GetProperties on a session object will only return properties in
// the session property group.
hr = _pUtilProp->SetProperties(
cPropertySets,
rgPropertySets,
PROPSET_COMMAND
);
if( (DB_E_ERRORSOCCURRED == hr) || (DB_S_ERRORSOCCURRED == hr) )
// check if a required property could not be set
{
ULONG i, j;
for(i = 0; i < cPropertySets; i++)
for(j = 0; j < rgPropertySets[i].cProperties; j++)
if( rgPropertySets[i].rgProperties[j].dwStatus !=
DBPROPSTATUS_OK )
if( rgPropertySets[i].rgProperties[j].dwOptions !=
DBPROPOPTIONS_OPTIONAL )
{
BAIL_ON_FAILURE( hr = DB_E_ERRORSOCCURRED );
}
else
fWarning = TRUE;
// if we get here, then there was all required properties were set
// successfully. However, hr could still be DB_ERRORSOCCURRED if all
// properties were optional and all of them could not be set. This
// condition is not an error for OpenRowset as noted in the comment
// above. Hence reset hr to S_OK.
hr = S_OK;
}
// we still need to catch other errors like E_INAVLIDARG
BAIL_ON_FAILURE(hr);
hr = SetSearchPrefs();
BAIL_ON_FAILURE( hr );
//
// Create RowProvider object to pass to rowset code
//
pbMultiValue = (BOOL *)AllocADsMem(sizeof(BOOL));
pAttrs = (LPWSTR *)AllocADsMem(cAttrs * sizeof(LPWSTR));
if( pAttrs )
pAttrs[0] = AllocADsStr(L"ADsPath");
if( !pAttrs || !pAttrs[0] || !pbMultiValue )
BAIL_ON_FAILURE( hr=E_OUTOFMEMORY );
*pbMultiValue = FALSE;
_pDSSearch->AddRef();
// Is this an NDS path? If so, set filter appropriately. Fix for #286560.
WCHAR lpszProgId[MAX_PATH];
BOOL fIsNds;
hr = CopyADsProgId(pTableID->uName.pwszName, lpszProgId);
BAIL_ON_FAILURE( hr );
if( !wcscmp(L"NDS", lpszProgId) )
fIsNds = TRUE;
else
fIsNds = FALSE;
hr = CRowProvider::CreateRowProvider(
_pDSSearch,
NULL,
pAttrs,
cAttrs,
cColumns,
prgInfo,
pStringBuffer,
IID_IRowProvider,
pbMultiValue,
TRUE,
pCreds,
(void **) &pRowProvider
);
BAIL_ON_FAILURE( hr );
//
// RowProvider responsible for deallocation
//
pbMultiValue = NULL;
hr= CRowset::CreateRowset(
pRowProvider,
(LPUNKNOWN)(IAccessor FAR *)this ,
this,
NULL,
cPropertySets,
rgPropertySets,
0,
NULL,
TRUE, // ADsPath is requested
FALSE, // not all attributes are requested
riid,
ppRowset
);
BAIL_ON_FAILURE( hr );
error:
if( _pDSSearch ) {
_pDSSearch->Release();
_pDSSearch = NULL;
}
if( pRowProvider )
pRowProvider->Release();
if( prgInfo )
_pIMalloc->Free(prgInfo);
if( pStringBuffer )
_pIMalloc->Free(pStringBuffer);
if( pbMultiValue )
FreeADsMem(pbMultiValue);
if (pAttrs)
{
for (i = 0; i < cAttrs; i++)
{
if (pAttrs[i])
FreeADsStr(pAttrs[0]);
}
FreeADsMem(pAttrs);
}
if( FAILED(hr) )
RRETURN( hr );
else if( fWarning )
RRETURN( DB_S_ERRORSOCCURRED );
else
RRETURN( hr );
}
//+---------------------------------------------------------------------------
//
// Function: CSessionObject::GetProperties
//
// Synopsis: Returns current settings of all properties in the DBPROPFLAGS_SESSION property
// group
//
// Arguments:
// cPropertySets count of restiction guids
// rgPropertySets restriction guids
// pcProperties count of properties returned
// prgProperties property information returned
//
// Returns:
// S_OK Session Object Interface returned
// E_INVALIDARG pcProperties or prgPropertyInfo was NULL
// E_OUTOFMEMORY Out of memory
//
//
// Modifies:
//
// History: 08-28-96 ShankSh Created.
//
//----------------------------------------------------------------------------
STDMETHODIMP
CSessionObject::GetProperties(
ULONG cPropIDSets,
const DBPROPIDSET rgPropIDSets[],
ULONG * pcPropSets,
DBPROPSET ** pprgPropSets
)
{
//
// Asserts
//
ADsAssert(_pUtilProp);
//
// Check in-params and NULL out-params in case of error
//
HRESULT hr = _pUtilProp->GetPropertiesArgChk(
cPropIDSets,
rgPropIDSets,
pcPropSets,
pprgPropSets,
PROPSET_SESSION);
if( FAILED(hr) )
RRETURN( hr );
//
// Just pass this call on to the utility object that manages our properties
//
RRETURN( _pUtilProp->GetProperties(
cPropIDSets,
rgPropIDSets,
pcPropSets,
pprgPropSets,
PROPSET_SESSION) );
}
//+---------------------------------------------------------------------------
//
// Function: CSessionObject::SetProperties
//
// Synopsis: Set properties in the DBPROPFLAGS_SESSION property group
//
// Arguments:
// cProperties
// rgProperties
//
// Returns:
// S_OK Session Object Interface returned
// E_INVALIDARG pcProperties or prgPropertyInfo was NULL
// E_OUTOFMEMORY Out of memory
//
// Modifies:
//
// History: 08-28-96 ShankSh Created.
//
//----------------------------------------------------------------------------
STDMETHODIMP
CSessionObject::SetProperties(
ULONG cPropertySets,
DBPROPSET rgPropertySets[]
)
{
//
// Asserts
//
ADsAssert(_pUtilProp);
//
// Just pass this call on to the utility object that manages our properties
//
RRETURN( _pUtilProp->SetProperties(
cPropertySets,
rgPropertySets,
PROPSET_SESSION) );
}
//+---------------------------------------------------------------------------
//
// Function: CSessionObject::CreateCommand
//
// Synopsis: Creates a brand new command and returns requested interface
//
// Arguments:
// pUnkOuter outer Unknown
// riid, IID desired
// ppCommand ptr to interface
//
//
// Returns:
// S_OK Command Object Interface returned
// E_INVALIDARG ppCommand was NULL
// E_NOINTERFACE IID not supported
//
// Modifies:
//
// History: 08-28-96 ShankSh Created.
//
//----------------------------------------------------------------------------
STDMETHODIMP
CSessionObject::CreateCommand(
IUnknown * pUnkOuter,
REFIID riid,
IUnknown ** ppCommand
)
{
CCommandObject* pCommand = NULL;
HRESULT hr;
//
// check in-params and NULL out-params in case of error
//
if( ppCommand )
*ppCommand = NULL;
else
RRETURN( E_INVALIDARG );
if( pUnkOuter )//&& !InlineIsEqualGUID(riid, IID_IUnknown) )
RRETURN( DB_E_NOAGGREGATION );
//
// open a CCommand object
//
pCommand = new CCommandObject(pUnkOuter);
if( !pCommand )
RRETURN( E_OUTOFMEMORY );
//
// initialize the object
//
if( !pCommand->FInit(this, _Credentials) ) {
delete pCommand;
RRETURN( E_OUTOFMEMORY );
}
//
// get requested interface pointer on DBSession
//
hr = pCommand->QueryInterface(riid, (void **)ppCommand);
if( FAILED( hr ) ) {
delete pCommand;
RRETURN( hr );
}
pCommand->Release();
//
// all went well
//
RRETURN( S_OK );
}
//+---------------------------------------------------------------------------
//
// Function: CSessionObject::GetDefaultColumnInfo
//
// Synopsis:
//
// Arguments:
//
// Returns:
//
//
// Modifies:
//
// History: 08-28-96 ShankSh Created.
//
//----------------------------------------------------------------------------
STDMETHODIMP
CSessionObject::GetDefaultColumnInfo(
ULONG * pcColumns,
DBCOLUMNINFO ** prgInfo,
OLECHAR ** ppStringBuffer
)
{
//
// Asserts
//
ADsAssert(_pIMalloc);
ADsAssert(pcColumns);
ADsAssert(prgInfo);
ADsAssert(ppStringBuffer);
//
// Allcoate memory for the Bookmark and ADsPath column
//
*prgInfo = (DBCOLUMNINFO*)_pIMalloc->Alloc(2 * sizeof(DBCOLUMNINFO));
*ppStringBuffer = (WCHAR*)_pIMalloc->Alloc((wcslen(L"ADsPath") + 1) * sizeof(WCHAR));
//
// Free memory on a failure
//
if( !(*prgInfo) )
RRETURN( E_OUTOFMEMORY );
if( !(*ppStringBuffer) ) {
_pIMalloc->Free(*prgInfo);
*prgInfo = NULL;
RRETURN( E_OUTOFMEMORY );
}
//
// Initialize the memory
//
ZeroMemory(*prgInfo, 2 * sizeof(DBCOLUMNINFO));
ZeroMemory(*ppStringBuffer, (wcslen(L"ADsPath") + 1) * sizeof(WCHAR));
wcscpy(*ppStringBuffer, OLESTR("ADsPath"));
//
// Fill up the Bookmark column
//
*pcColumns = 2;
(*prgInfo)[0].pwszName = NULL;
(*prgInfo)[0].pTypeInfo = NULL;
(*prgInfo)[0].iOrdinal = 0;
(*prgInfo)[0].ulColumnSize = sizeof(ULONG);
(*prgInfo)[0].wType = DBTYPE_UI4;
(*prgInfo)[0].bPrecision = 10;
(*prgInfo)[0].bScale = (BYTE) ~ 0;
(*prgInfo)[0].columnid.eKind = DBKIND_GUID_PROPID;
(*prgInfo)[0].columnid.uGuid.guid = DBCOL_SPECIALCOL;
(*prgInfo)[0].columnid.uName.ulPropid = 2;
(*prgInfo)[0].dwFlags = DBCOLUMNFLAGS_ISBOOKMARK |
DBCOLUMNFLAGS_ISFIXEDLENGTH;
//
// Fill up the ADsPath column
//
(*prgInfo)[1].pwszName = *ppStringBuffer;
(*prgInfo)[1].pTypeInfo = NULL;
(*prgInfo)[1].iOrdinal = 1;
(*prgInfo)[1].ulColumnSize = (ULONG)256;
(*prgInfo)[1].wType = DBTYPE_WSTR|DBTYPE_BYREF;
(*prgInfo)[1].bPrecision = (BYTE) ~ 0;
(*prgInfo)[1].bScale = (BYTE) ~ 0;
(*prgInfo)[1].columnid.eKind = DBKIND_NAME;
(*prgInfo)[1].columnid.uName.pwszName = *ppStringBuffer;
(*prgInfo)[1].columnid.uGuid.guid = GUID_NULL;
(*prgInfo)[1].dwFlags = DBCOLUMNFLAGS_ISNULLABLE;
RRETURN( S_OK );
}
#if (!defined(BUILD_FOR_NT40))
//IBindResource::Bind
//+---------------------------------------------------------------------------
//
// Function: CSessionObject::CSessionObject
//
// Synopsis: Constructor
//
// Arguments:
//
// Returns:
//
// Modifies:
//
// History: 09-17-1998 mgorti Created.
//
//----------------------------------------------------------------------------
HRESULT CSessionObject::Bind(
IUnknown * punkOuter,
LPCOLESTR pwszURL,
DBBINDURLFLAG dwBindFlags,
REFGUID rguid,
REFIID riid,
IAuthenticate * pAuthenticate,
DBIMPLICITSESSION * pImplSession,
DWORD * pdwBindStatus,
IUnknown ** ppUnk
)
{
HRESULT hr = S_OK;
CComBSTR bstrNewURL;
TRYBLOCK
//Initialize return arguments
if (pdwBindStatus)
*pdwBindStatus = 0;
if (ppUnk)
*ppUnk = NULL;
if (pImplSession)
pImplSession->pSession = NULL;
//if caller passed a null value for dwBindFlags,
//get them from initialization properties.
if (dwBindFlags == 0)
dwBindFlags = BindFlagsFromDbProps();
//Generic argument validation
hr = ValidateBindArgs(punkOuter,
pwszURL,
dwBindFlags,
rguid,
riid,
pAuthenticate,
pImplSession,
pdwBindStatus,
ppUnk);
BAIL_ON_FAILURE(hr);
//Fill in the pImplSession struct.
if (pImplSession)
{
// Our session doesn't support aggregation.
if (pImplSession->pUnkOuter != NULL)
BAIL_ON_FAILURE(hr = DB_E_NOAGGREGATION);
hr = QueryInterface(*pImplSession->piid,
(void**)&(pImplSession->pSession));
if (FAILED(hr))
BAIL_ON_FAILURE(hr = E_NOINTERFACE );
}
//Specific validation checks
//We are currently a read-only provider
if (dwBindFlags & DBBINDURLFLAG_WRITE)
BAIL_ON_FAILURE(hr = DB_E_READONLY);
//We currently don't support aggregation
if (punkOuter != NULL)
BAIL_ON_FAILURE (hr = DB_E_NOAGGREGATION);
//We don't support the following flags
if (dwBindFlags & DBBINDURLFLAG_ASYNCHRONOUS)
BAIL_ON_FAILURE(hr = DB_E_ASYNCNOTSUPPORTED);
if (dwBindFlags & DBBINDURLFLAG_OUTPUT ||
dwBindFlags & DBBINDURLFLAG_RECURSIVE ||
dwBindFlags & DBBINDURLFLAG_DELAYFETCHSTREAM ||
dwBindFlags & DBBINDURLFLAG_DELAYFETCHCOLUMNS)
BAIL_ON_FAILURE(hr = E_INVALIDARG);
//Now Try to Bind.
if (InlineIsEqualGUID(rguid, DBGUID_ROW) ||
InlineIsEqualGUID(rguid, DBGUID_ROWSET))
{
//If the URL is not absolute, build the absolute URL
//using the DBPROP_INIT_PROVIDERSTRING.
if (! bIsAbsoluteURL(pwszURL))
{
hr = BuildAbsoluteURL (pwszURL, bstrNewURL);
if (FAILED(hr))
BAIL_ON_FAILURE(hr = E_INVALIDARG);
}
else
bstrNewURL = pwszURL;
if ( InlineIsEqualGUID(rguid, DBGUID_ROW) )
{
hr = BindToRow(punkOuter,
(PWCHAR)bstrNewURL,
pAuthenticate,
dwBindFlags,
riid,
ppUnk);
}
else
{
hr = BindToRowset(punkOuter,
(PWCHAR)bstrNewURL,
pAuthenticate,
dwBindFlags,
riid,
ppUnk);
}
BAIL_ON_FAILURE(hr);
}
else if (InlineIsEqualGUID(rguid, DBGUID_DSO))
{
ADsAssert(_pDSO);
hr = BindToDataSource(
punkOuter,
pwszURL,
pAuthenticate,
dwBindFlags,
riid,
ppUnk
);
BAIL_ON_FAILURE (hr);
}
else if (InlineIsEqualGUID(rguid, DBGUID_SESSION))
{
hr = QueryInterface(riid, (void**)ppUnk);
BAIL_ON_FAILURE (hr);
}
else
BAIL_ON_FAILURE(hr = E_INVALIDARG);
//Fix for bug Raid-X5#83386 - spec change
//If caller specified any DENY semantics,
//set warning status and return value, since
//we don't support these.
if (dwBindFlags & DBBINDURLFLAG_SHARE_DENY_READ ||
dwBindFlags & DBBINDURLFLAG_SHARE_DENY_WRITE ||
dwBindFlags & DBBINDURLFLAG_SHARE_EXCLUSIVE)
{
if (pdwBindStatus)
*pdwBindStatus = DBBINDURLSTATUS_S_DENYNOTSUPPORTED;
BAIL_ON_FAILURE (hr = DB_S_ERRORSOCCURRED);
}
CATCHBLOCKBAIL(hr)
error:
RRETURN(hr);
}
#endif
//+---------------------------------------------------------------------------
//
// Function: CSessionObject::CSessionObject
//
// Synopsis: Constructor
//
// Arguments:
//
// Returns:
//
// Modifies:
//
// History: 08-28-96 ShankSh Created.
//
//----------------------------------------------------------------------------
CSessionObject::CSessionObject(
LPUNKNOWN pUnkOuter
)
{
//
// Initialize simple member vars
//
_pUnkOuter = pUnkOuter ? pUnkOuter : (IGetDataSource FAR *)this;
_cCommandsOpen = 0;
_pUtilProp = NULL;
_pDSSearch = NULL;
_pIMalloc = NULL;
_pDSO = NULL;
ENLIST_TRACKING(CSessionObject);
}
//+---------------------------------------------------------------------------
//
// Function: CSessionObject::~CSessionObject
//
// Synopsis: Destructor
//
// Arguments:
//
// Returns:
//
// Modifies:
//
// History: 08-28-96 ShankSh Created.
//
//----------------------------------------------------------------------------
CSessionObject::~CSessionObject( )
{
//
// Free properties management object
//
delete _pUtilProp;
if( _pIMalloc )
_pIMalloc->Release();
if( _pDSO ) {
_pDSO->DecrementOpenSessions();
_pDSO->Release();
}
}
//+---------------------------------------------------------------------------
//
// Function: CSessionObject::FInit
//
// Synopsis: Initialize the session Object
//
// Arguments:
//
// Returns:
// TRUE Initialization succeeded
// FALSE Initialization failed
//
// Modifies:
//
// History: 08-28-96 ShankSh Created.
//
//----------------------------------------------------------------------------
BOOL CSessionObject::FInit(
CDSOObject * pDSO,
CCredentials & Credentials
)
{
HRESULT hr;
//
// Asserts
//
ADsAssert(pDSO);
ADsAssert(&Credentials);
//
// Allocate properties management object
//
_pUtilProp = new CUtilProp();
if( !_pUtilProp )
return FALSE;
hr = _pUtilProp->FInit(&Credentials);
BAIL_ON_FAILURE(hr);
//
// IMalloc->Alloc is the way we have to allocate memory for out parameters
//
hr = CoGetMalloc(MEMCTX_TASK, &_pIMalloc);
BAIL_ON_FAILURE(hr);
//
// Establish parent object pointer
//
_pDSO = pDSO;
_Credentials = Credentials;
_pDSO->AddRef();
_pDSO->IncrementOpenSessions();
return( TRUE );
error:
RRETURN( FALSE );
}
STDMETHODIMP
CSessionObject::QueryInterface(REFIID iid, LPVOID FAR* ppv)
{
if( ppv == NULL )
RRETURN( E_INVALIDARG );
if( IsEqualIID(iid, IID_IUnknown) ) {
*ppv = (IGetDataSource FAR *) this;
}
else if( IsEqualIID(iid, IID_IGetDataSource) ) {
*ppv = (IGetDataSource FAR *) this;
}
else if( IsEqualIID(iid, IID_IOpenRowset) ) {
*ppv = (IOpenRowset FAR *) this;
}
else if( IsEqualIID(iid, IID_ISessionProperties) ) {
*ppv = (ISessionProperties FAR *) this;
}
else if( IsEqualIID(iid, IID_IDBCreateCommand) ) {
*ppv = (IDBCreateCommand FAR *) this;
}
#if (!defined(BUILD_FOR_NT40))
else if( IsEqualIID(iid, IID_IBindResource) ) {
*ppv = (IBindResource FAR *) this;
}
#endif
else {
*ppv = NULL;
return E_NOINTERFACE;
}
AddRef();
return NOERROR;
}
#if (!defined(BUILD_FOR_NT40))
//+---------------------------------------------------------------------------
//
// Function: CSessionObject::ValidateBindArgs
//
// Synopsis: Validates IBindResource::Bind function arguments.
//
//----------------------------------------------------------------------------
HRESULT CSessionObject::ValidateBindArgs(
IUnknown * punkOuter,
LPCOLESTR pwszURL,
DBBINDURLFLAG dwBindFlags,
REFGUID rguid,
REFIID riid,
IAuthenticate * pAuthenticate,
DBIMPLICITSESSION * pImplSession,
DWORD * pdwBindStatus,
IUnknown ** ppUnk
)
{
//General validation checks.
if (pwszURL == NULL || InlineIsEqualGUID(rguid,GUID_NULL) ||
InlineIsEqualGUID(riid, GUID_NULL) || ppUnk == NULL )
RRETURN(E_INVALIDARG);
if (pImplSession &&
(pImplSession->pUnkOuter == NULL || pImplSession->piid == NULL))
RRETURN(E_INVALIDARG);
if (punkOuter && !InlineIsEqualGUID(riid, IID_IUnknown))
RRETURN(DB_E_NOAGGREGATION);
if (pImplSession && pImplSession->pUnkOuter &&
pImplSession->piid &&
!InlineIsEqualGUID(*pImplSession->piid, IID_IUnknown))
RRETURN(DB_E_NOAGGREGATION);
if (dwBindFlags & DBBINDURLFLAG_RECURSIVE)
{
//if DBBINDURLFLAG_RECURSIVE is set, at least one of the SHARE_DENY
//flags must have been set.
if (! ( (dwBindFlags & DBBINDURLFLAG_SHARE_DENY_READ) ||
(dwBindFlags & DBBINDURLFLAG_SHARE_DENY_WRITE) ||
(dwBindFlags & DBBINDURLFLAG_SHARE_EXCLUSIVE)
)
)
RRETURN(E_INVALIDARG);
}
if (!(dwBindFlags & DBBINDURLFLAG_READ) &&
!(dwBindFlags & DBBINDURLFLAG_WRITE) ) {
// Must have either read or write access:
RRETURN(E_INVALIDARG);
}
if (InlineIsEqualGUID(rguid, DBGUID_DSO) &&
!((dwBindFlags & DBBINDURLFLAG_READ) ||
(dwBindFlags & DBBINDURLFLAG_ASYNCHRONOUS) ||
(dwBindFlags & DBBINDURLFLAG_WAITFORINIT)
)
)
//if object type is DataSource, only the above flags are allowed
RRETURN(E_INVALIDARG);
if (InlineIsEqualGUID(rguid, DBGUID_SESSION) &&
! (dwBindFlags == DBBINDURLFLAG_READ))
//if object type is Session, only DBBINDURLFLAG_READ is allowed
RRETURN(E_INVALIDARG);
if (InlineIsEqualGUID(rguid, DBGUID_ROWSET) &&
((dwBindFlags & DBBINDURLFLAG_DELAYFETCHCOLUMNS) ||
(dwBindFlags & DBBINDURLFLAG_DELAYFETCHSTREAM)
)
)
//if object type is Rowset, DELAYFETCHCOLUMNS and DELAYFETCHSTREAM
//flags are disallowed.
RRETURN ( E_INVALIDARG );
if (InlineIsEqualGUID(rguid, DBGUID_STREAM) &&
((dwBindFlags & DBBINDURLFLAG_DELAYFETCHCOLUMNS) ||
(dwBindFlags & DBBINDURLFLAG_DELAYFETCHSTREAM)
)
)
//if object type is Stream, DELAYFETCHCOLUMNS and
//DELAYFETCHSTREAM flags are disallowed.
RRETURN(E_INVALIDARG);
RRETURN(S_OK);
}
//+---------------------------------------------------------------------------
//
// Function: CSessionObject::BindToRow
//
// Synopsis: Given a URL, binds to that row object and returns the requested
// interface.
//
//----------------------------------------------------------------------------
HRESULT
CSessionObject::BindToRow(
IUnknown *punkOuter,
LPCOLESTR pwszURL,
IAuthenticate *pAuthenticate,
DWORD dwBindFlags,
REFIID riid,
IUnknown** ppUnk
)
{
CComObject<CRow> *pRow = NULL;
auto_rel<IUnknown> pSession;
auto_rel<IRow> pRowDelete;
HRESULT hr = S_OK;
hr = CComObject<CRow>::CreateInstance(&pRow);
if (FAILED(hr))
BAIL_ON_FAILURE(hr = E_OUTOFMEMORY);
//To make sure we delete the row object in
//case we encounter errors after this point.
//Note: this version of auto_rel doesn't addref on assignment.
pRowDelete = pRow;
pRowDelete->AddRef();
hr = QueryInterface(__uuidof(IUnknown), (void **)&pSession);
if (FAILED(hr))
BAIL_ON_FAILURE(hr = E_FAIL);
//Initialize row and bind it to a Directory Object.
hr = pRow->Initialize((PWSTR)pwszURL,
(IUnknown *)pSession,
pAuthenticate,
dwBindFlags,
FALSE, // not a tearoff
FALSE, // don't get column info. from rowset
&_Credentials,
true);
if (FAILED(hr))
{
if (INVALID_CREDENTIALS_ERROR(hr))
{
BAIL_ON_FAILURE(hr = DB_SEC_E_PERMISSIONDENIED);
}
else
{
BAIL_ON_FAILURE(hr = DB_E_NOTFOUND);
}
}
hr = pRow->QueryInterface(riid, (void**)ppUnk);
if (FAILED(hr))
BAIL_ON_FAILURE (hr = E_NOINTERFACE);
error:
RRETURN ( hr );
}
//+---------------------------------------------------------------------------
//
// Function: CSessionObject::BindToRowset
//
// Synopsis: Given a URL, binds to a rowset object that has all its child
// nodes as rows and returns the requested interface on the rowset.
//
//----------------------------------------------------------------------------
HRESULT
CSessionObject::BindToRowset(
IUnknown *pUnkOuter,
LPCOLESTR pwszURL,
IAuthenticate *pAuthenticate,
DWORD dwBindFlags,
REFIID riid,
IUnknown** ppUnk
)
{
HRESULT hr;
DWORD fAuthFlags;
DBID tableID;
tableID.eKind = DBKIND_NAME;
tableID.uName.pwszName = (LPWSTR) pwszURL;
//Create the rowset.
// Fix for 351040. First try explicit credentials, then session object's
// credentials, then default credentials.
if(pAuthenticate)
{
CCredentials creds;
hr = GetCredentialsFromIAuthenticate(pAuthenticate, creds);
if (FAILED(hr))
BAIL_ON_FAILURE(hr = E_INVALIDARG);
fAuthFlags = creds.GetAuthFlags();
creds.SetAuthFlags(fAuthFlags |
ADS_SECURE_AUTHENTICATION);
hr = OpenRowsetWithCredentials(pUnkOuter, &tableID, NULL, riid,
0, NULL, &creds, ppUnk);
}
if( (!pAuthenticate) || (INVALID_CREDENTIALS_ERROR(hr)) )
// try credentials in session object
hr = OpenRowset(pUnkOuter, &tableID, NULL, riid, 0, NULL, ppUnk);
if(INVALID_CREDENTIALS_ERROR(hr))
// try default credentials
{
CCredentials creds; // default credentials
fAuthFlags = creds.GetAuthFlags();
creds.SetAuthFlags(fAuthFlags |
ADS_SECURE_AUTHENTICATION);
hr = OpenRowsetWithCredentials(pUnkOuter, &tableID, NULL, riid,
0, NULL, &creds, ppUnk);
}
BAIL_ON_FAILURE(hr);
RRETURN ( S_OK );
error:
RRETURN ( hr );
}
//+---------------------------------------------------------------------------
//
// Function: CSessionObject::BindFlagsFromDbProps
//
// Synopsis: Synthesizes bind flags from initialization properties
// DBPROP_INIT_MODE and DBPROP_INIT_BINDFLAGS
//
//----------------------------------------------------------------------------
DWORD CSessionObject::BindFlagsFromDbProps()
{
HRESULT hr = S_OK;
auto_rel<IDBProperties> pDBProp;
ULONG i, j, cPropertySets = 0;
DWORD dwMode = 0, dwBindFlags = 0, dwBindFlagProp = 0;
DWORD dwResult = 0;
hr = GetDataSource(__uuidof(IDBProperties), (IUnknown **)&pDBProp);
BAIL_ON_FAILURE(hr);
DBPROPID propids[2];
propids[0] = DBPROP_INIT_MODE;
propids[1] = DBPROP_INIT_BINDFLAGS;
DBPROPIDSET rgPropertyIDSets[1];
rgPropertyIDSets[0].rgPropertyIDs = propids;
rgPropertyIDSets[0].cPropertyIDs = 2;
rgPropertyIDSets[0].guidPropertySet = DBPROPSET_DBINIT;
DBPROPSET *prgPropertySets;
hr = pDBProp->GetProperties(
1,
rgPropertyIDSets,
&cPropertySets,
&prgPropertySets);
if (hr == DB_E_ERRORSOCCURRED)
BAIL_ON_FAILURE(hr);
for (i = 0; i < cPropertySets; i++)
{
for (j = 0; j < prgPropertySets[i].cProperties; j++)
{
DBPROP *pProp = &prgPropertySets[i].rgProperties[j];
ADsAssert(pProp);
if (pProp->dwStatus == S_OK &&
pProp->dwPropertyID == DBPROP_INIT_MODE)
dwMode = V_I4(&pProp->vValue);
else if (pProp->dwStatus == S_OK &&
pProp->dwPropertyID == DBPROP_INIT_BINDFLAGS)
dwBindFlagProp = V_I4(&pProp->vValue);
else
continue;
}
}
//Now extract bind flags from dwMode and dwBindFlagProp
{
DWORD dwModeMask =
DB_MODE_READ |
DB_MODE_WRITE |
DB_MODE_READWRITE |
DB_MODE_SHARE_DENY_READ |
DB_MODE_SHARE_DENY_WRITE |
DB_MODE_SHARE_EXCLUSIVE |
DB_MODE_SHARE_DENY_NONE;
dwResult |= dwMode & dwModeMask;
if ( dwBindFlagProp & DB_BINDFLAGS_DELAYFETCHCOLUMNS ) {
dwBindFlags |= DBBINDURLFLAG_DELAYFETCHCOLUMNS;
}
if ( dwBindFlagProp & DB_BINDFLAGS_DELAYFETCHSTREAM ) {
dwBindFlags |= DBBINDURLFLAG_DELAYFETCHSTREAM;
}
if ( dwBindFlagProp & DB_BINDFLAGS_RECURSIVE ) {
dwBindFlags |= DBBINDURLFLAG_RECURSIVE;
}
if ( dwBindFlagProp & DB_BINDFLAGS_OUTPUT ) {
dwBindFlags |= DBBINDURLFLAG_OUTPUT;
}
dwResult |= dwBindFlagProp | dwBindFlags;
}
error:
for (i = 0; i < cPropertySets; i++)
{
for (j = 0; j < prgPropertySets[i].cProperties; j++)
{
DBPROP *pProp = &(prgPropertySets[i].rgProperties[j]);
ADsAssert(pProp);
FreeDBID(&pProp->colid);
VariantClear(&pProp->vValue);
}
CoTaskMemFree(prgPropertySets[i].rgProperties);
}
CoTaskMemFree(prgPropertySets);
RRETURN ( dwResult );
}
//+---------------------------------------------------------------------------
//
// Function: CSessionObject::BindToDataSource
//
// Synopsis: Initializes the DataSource object if necessary, Sets
// DBPROP_INIT_PROVIDERSTRING property and returns the requested
// interface on the datasource.
//
//----------------------------------------------------------------------------
HRESULT
CSessionObject::BindToDataSource(
IUnknown *pUnkOuter,
LPCOLESTR pwszURL,
IAuthenticate *pAuthenticate,
DWORD dwBindFlags,
REFIID riid,
IUnknown** ppUnk
)
{
HRESULT hr = S_OK;
auto_rel<IDBProperties> pDBProperties;
DBPROP props[1];
DBPROPSET rgPropertySets[1];
CComBSTR bstrURL(pwszURL);
//Initialize DBPROP_INIT_PROVIDERSTRING only if the
//URL is absolute.
if (bIsAbsoluteURL (pwszURL))
{
// Check if the datasource has already been initialized.
if (_pDSO->IsInitialized())
BAIL_ON_FAILURE(hr = DB_E_ALREADYINITIALIZED);
props[0].dwPropertyID = DBPROP_INIT_PROVIDERSTRING;
props[0].dwOptions = DBPROPOPTIONS_OPTIONAL;
props[0].vValue.vt = VT_BSTR;
props[0].vValue.bstrVal = (PWCHAR)bstrURL;
rgPropertySets[0].rgProperties = props;
rgPropertySets[0].cProperties = 1;
rgPropertySets[0].guidPropertySet = DBPROPSET_DBINIT;
hr = GetDataSource(
__uuidof(IDBProperties),
(IUnknown **)&pDBProperties
);
BAIL_ON_FAILURE(hr);
hr = pDBProperties->SetProperties(1, rgPropertySets);
BAIL_ON_FAILURE(hr);
}
// If consumer doesn't specify DBBINDURLFLAG_WAITFORINIT, it
// means consumer wants an initialized DSO
//
if (! (dwBindFlags & DBBINDURLFLAG_WAITFORINIT))
{
auto_rel<IDBInitialize> pDBInitialize;
hr = GetDataSource(__uuidof(IDBInitialize), (IUnknown **)&pDBInitialize);
BAIL_ON_FAILURE(hr);
hr = pDBInitialize->Initialize();
BAIL_ON_FAILURE(hr);
}
//Return the requested interface on the DSO.
hr = GetDataSource(riid, ppUnk);
BAIL_ON_FAILURE(hr);
error:
RRETURN ( hr );
}
//+---------------------------------------------------------------------------
//
// Function: CSessionObject::BuildAbsoluteURL
//
// Synopsis: Given a relative URL, builds absolute URL using the relative URL
// and the property DBPROP_INIT_PROVIDERSTRING.
//
//----------------------------------------------------------------------------
HRESULT
CSessionObject::BuildAbsoluteURL(
CComBSTR bstrLeaf,
CComBSTR& bstrAbsoluteURL
)
{
HRESULT hr = S_OK;
auto_rel<IDBProperties> pDBProp;
auto_rel<IADsPathname> pPathParent;
auto_rel<IADsPathname> pPathLeaf;
ULONG cPropertySets = 0;
long i, j, cElements = 0;
CComBSTR bstrParent;
DBPROPSET* prgPropertySets = NULL;
DBPROPID propids[1];
DBPROPIDSET rgPropertyIDSets[1];
hr = GetDataSource(__uuidof(IDBProperties), (IUnknown **)&pDBProp);
BAIL_ON_FAILURE(hr);
propids[0] = DBPROP_INIT_PROVIDERSTRING;
rgPropertyIDSets[0].rgPropertyIDs = propids;
rgPropertyIDSets[0].cPropertyIDs = 1;
rgPropertyIDSets[0].guidPropertySet = DBPROPSET_DBINIT;
hr = pDBProp->GetProperties(
1,
rgPropertyIDSets,
&cPropertySets,
&prgPropertySets);
if (SUCCEEDED(hr) && cPropertySets == 1)
{
ADsAssert(prgPropertySets != NULL);
ADsAssert(prgPropertySets[0].rgProperties != NULL);
DBPROP* pProp = & (prgPropertySets[0].rgProperties[0]);
bstrParent = pProp->vValue.bstrVal;
}
// Build Absolute Path from Parent and leaf.
hr = CPathname::CreatePathname(
__uuidof(IADsPathname),
(void **)&pPathParent
);
BAIL_ON_FAILURE(hr);
hr = pPathParent->Set(bstrParent, ADS_SETTYPE_FULL);
BAIL_ON_FAILURE(hr);
if (bstrLeaf.Length() > 0)
{
hr = CPathname::CreatePathname(
__uuidof(IADsPathname),
(void **)&pPathLeaf
);
BAIL_ON_FAILURE(hr);
hr = pPathLeaf->Set(bstrLeaf, ADS_SETTYPE_DN);
BAIL_ON_FAILURE(hr);
hr = pPathLeaf->GetNumElements(&cElements);
BAIL_ON_FAILURE(hr);
//Add leaf elements in reverse order.
//Ex: if bstrLeaf = "CN=Administrator,CN=Users",
//we add CN=Users first.
for (i = cElements-1; i >= 0; i--)
{
CComBSTR bstrElement;
hr = pPathLeaf->GetElement(i, &bstrElement);
BAIL_ON_FAILURE(hr);
hr = pPathParent->AddLeafElement(bstrElement);
BAIL_ON_FAILURE(hr);
}
}
//Read back the fully built path name
hr = pPathParent->Retrieve(ADS_FORMAT_X500, &bstrAbsoluteURL);
BAIL_ON_FAILURE(hr);
error:
// Free memory allocated by GetProperties
for (i = 0; i < cPropertySets; i++)
{
for (j = 0; j < prgPropertySets[i].cProperties; j++)
{
DBPROP *pProp = &(prgPropertySets[i].rgProperties[j]);
ADsAssert(pProp);
// We should free the DBID in pProp, but we know that
// GetProperties always returns DB_NULLID and FreeDBID doesn't
// handle DB_NULLID. So, DBID is not freed here.
VariantClear(&pProp->vValue);
}
CoTaskMemFree(prgPropertySets[i].rgProperties);
}
CoTaskMemFree(prgPropertySets);
RRETURN(hr);
}
extern PROUTER_ENTRY g_pRouterHead;
extern CRITICAL_SECTION g_csRouterHeadCritSect;
//+---------------------------------------------------------------------------
//
// Function: CSessionObject::bIsAbsoluteURL
//
// Synopsis: If the given URL starts with any of the ADS provider prefixes,
// returns true. Returns false otherwise.
//
//----------------------------------------------------------------------------
bool
CSessionObject::bIsAbsoluteURL( LPCOLESTR pwszURL)
{
if (pwszURL == NULL)
return false;
//
// Make sure the router has been initialized
//
EnterCriticalSection(&g_csRouterHeadCritSect);
if (!g_pRouterHead) {
g_pRouterHead = InitializeRouter();
}
LeaveCriticalSection(&g_csRouterHeadCritSect);
for (PROUTER_ENTRY pProvider = g_pRouterHead;
pProvider != NULL;
pProvider = pProvider->pNext)
{
if (pProvider->szProviderProgId == NULL)
continue;
size_t strSize = wcslen(pProvider->szProviderProgId);
if ( _wcsnicmp(pwszURL, pProvider->szProviderProgId, strSize) == 0 )
return true;
}
// Given URL doesn't start with any of the ADSI provider prefixes.
return false;
}
#endif
//-----------------------------------------------------------------------------
// SetSearchPrefs
//
// Sets ADSI search preferences on the property object.
//
//-----------------------------------------------------------------------------
STDMETHODIMP
CSessionObject::SetSearchPrefs(
void
)
{
PROPSET *pPropSet;
PADS_SEARCHPREF_INFO pSearchPref = NULL;
HRESULT hr = S_OK;
ULONG i;
//
// Asserts
//
ADsAssert(_pUtilProp);
ADsAssert(_pDSSearch);
pPropSet = _pUtilProp->GetPropSetFromGuid(DBPROPSET_ADSISEARCH);
if( !pPropSet || !pPropSet->cProperties )
RRETURN( S_OK );
pSearchPref = (PADS_SEARCHPREF_INFO) AllocADsMem(
pPropSet->cProperties *
sizeof(ADS_SEARCHPREF_INFO)
);
if( !pSearchPref )
BAIL_ON_FAILURE( hr=E_OUTOFMEMORY );
for (i=0; i<pPropSet->cProperties; i++) {
hr = _pUtilProp->GetSearchPrefInfo(
pPropSet->pUPropInfo[i].dwPropertyID,
&pSearchPref[i]
);
BAIL_ON_FAILURE( hr );
}
hr = _pDSSearch->SetSearchPreference(
pSearchPref,
pPropSet->cProperties
);
_pUtilProp->FreeSearchPrefInfo(pSearchPref, pPropSet->cProperties);
BAIL_ON_FAILURE( hr );
error:
if( pSearchPref )
FreeADsMem(pSearchPref);
RRETURN( hr );
}