//--------------------------------------------------------------------------- // // Microsoft Windows // Copyright (C) Microsoft Corporation, 1992 - 1995 // // File: cenumGroupCollection.cxx // // Contents: Windows NT 3.5 GroupCollection Enumeration Code // // History: //---------------------------------------------------------------------------- #include "winnt.hxx" #pragma hdrstop #define BASE_BUFFER_SIZE (4096 * sizeof(WCHAR)) CWinNTUserGroupsCollectionEnum::CWinNTUserGroupsCollectionEnum(): _ParentType(0), _ParentADsPath(NULL), _DomainName(NULL), _ServerName(NULL), _UserName(NULL), _pGlobalBuffer(NULL), _pLocalBuffer(NULL), _dwCurrent(0), _dwTotal(0), _dwGlobalTotal(0), _dwLocalTotal(0), _fIsDomainController(FALSE) { VariantInit(&_vFilter); } CWinNTUserGroupsCollectionEnum::~CWinNTUserGroupsCollectionEnum() { if (_ParentADsPath) ADsFreeString(_ParentADsPath); if (_DomainName) ADsFreeString(_DomainName); if (_ServerName) ADsFreeString(_ServerName); if (_UserName) ADsFreeString(_UserName); if (_pGlobalBuffer) NetApiBufferFree(_pGlobalBuffer); if (_pLocalBuffer) NetApiBufferFree(_pLocalBuffer); VariantClear(&_vFilter); } //+--------------------------------------------------------------------------- // // Function: CWinNTEnumVariant::Create // // Synopsis: // // Arguments: [pCollection] // [ppEnumVariant] // // Returns: HRESULT // // Modifies: // // History: 01-30-95 krishnag Created. // //---------------------------------------------------------------------------- HRESULT CWinNTUserGroupsCollectionEnum::Create( CWinNTUserGroupsCollectionEnum FAR* FAR* ppenumvariant, ULONG ParentType, BSTR ParentADsPath, BSTR DomainName, BSTR ServerName, BSTR UserName, VARIANT vFilter, CWinNTCredentials& Credentials ) { HRESULT hr = NOERROR; CWinNTUserGroupsCollectionEnum FAR* penumvariant = NULL; WCHAR szServer[MAX_PATH]; DSROLE_PRIMARY_DOMAIN_INFO_BASIC* pdomainInfo = NULL; DWORD dwResult = ERROR_SUCCESS; // // Should the checks below be assertions? // if (!ppenumvariant) return E_FAIL; if (ParentType != WINNT_DOMAIN_ID && ParentType != WINNT_COMPUTER_ID) return E_FAIL; *ppenumvariant = NULL; penumvariant = new CWinNTUserGroupsCollectionEnum(); if (!penumvariant) { hr = E_OUTOFMEMORY; BAIL_ON_FAILURE(hr); } penumvariant->_ParentType = ParentType; hr = ADsAllocString(ParentADsPath , &penumvariant->_ParentADsPath); BAIL_ON_FAILURE(hr); hr = ADsAllocString(DomainName, &penumvariant->_DomainName); BAIL_ON_FAILURE(hr); hr = ADsAllocString(ServerName, &penumvariant->_ServerName); BAIL_ON_FAILURE(hr); hr = ADsAllocString(UserName, &penumvariant->_UserName); BAIL_ON_FAILURE(hr); // // We currently copy the filter but do nothing with it!!! // hr = VariantCopy(&penumvariant->_vFilter, &vFilter); BAIL_ON_FAILURE(hr); // check if the server is a DC hr = MakeUncName(ServerName, szServer); BAIL_ON_FAILURE(hr); dwResult = DsRoleGetPrimaryDomainInformation( szServer, DsRolePrimaryDomainInfoBasic, // InfoLevel (PBYTE*)&pdomainInfo // pBuffer ); hr = HRESULT_FROM_WIN32(dwResult); BAIL_ON_FAILURE(hr); if( (pdomainInfo->MachineRole == DsRole_RoleBackupDomainController) || (pdomainInfo->MachineRole == DsRole_RolePrimaryDomainController) ) { penumvariant->_fIsDomainController = TRUE; } else penumvariant->_fIsDomainController = FALSE; hr = penumvariant->DoEnumeration(); BAIL_ON_FAILURE(hr); penumvariant->_Credentials = Credentials; hr = penumvariant->_Credentials.Ref(ServerName, DomainName, ParentType); BAIL_ON_FAILURE(hr); if ( pdomainInfo ) { DsRoleFreeMemory(pdomainInfo); } *ppenumvariant = penumvariant; RRETURN(hr); error: delete penumvariant; if ( pdomainInfo ) { DsRoleFreeMemory(pdomainInfo); } RRETURN_EXP_IF_ERR(hr); } HRESULT CWinNTUserGroupsCollectionEnum::DoEnumeration() { HRESULT hr; if (_ParentType == WINNT_DOMAIN_ID) { hr = DoGlobalEnumeration(); BAIL_ON_FAILURE(hr); hr = DoLocalEnumeration(); } else if (_ParentType == WINNT_COMPUTER_ID) { // We also want to try and get the global groups, // as this will be empty if the user does not belong to // any global groups. // We need to be careful on where we fail as we should // continue enumerating local groups for lots of cases. hr = DoGlobalEnumeration(); if ((hr == HRESULT_FROM_WIN32(ERROR_ACCESS_DENIED)) || (hr == HRESULT_FROM_WIN32(NERR_InvalidComputer)) || (hr == HRESULT_FROM_WIN32(NERR_UserNotFound)) || (hr == E_FAIL)) { hr = S_OK; _dwGlobalTotal = 0; } BAIL_ON_FAILURE(hr); hr = DoLocalEnumeration(); } else { hr = E_FAIL; } error: _dwTotal = _dwGlobalTotal + _dwLocalTotal; RRETURN(hr); } HRESULT CWinNTUserGroupsCollectionEnum::DoGlobalEnumeration() { HRESULT hr; DWORD dwStatus; DWORD dwRead = 0; WCHAR szServer[MAX_PATH]; LPGROUP_USERS_INFO_0 pGroupUserInfo = NULL; GROUP_INFO_2 *pGroupInfo2 = NULL; NET_API_STATUS nasStatus; hr = MakeUncName(_ServerName, szServer); BAIL_ON_FAILURE(hr); dwStatus = NetUserGetGroups(szServer, _UserName, 0, (LPBYTE*)&_pGlobalBuffer, MAX_PREFERRED_LENGTH, &dwRead, &_dwGlobalTotal); hr = HRESULT_FROM_WIN32(dwStatus); BAIL_ON_FAILURE(hr); if (dwRead != _dwGlobalTotal) hr = E_FAIL; if (dwRead == 1) { // // Check if it is the none group - dont want that // pGroupUserInfo = (LPGROUP_USERS_INFO_0)_pGlobalBuffer; if ( pGroupUserInfo->grui0_name && (FALSE == _fIsDomainController) && (1 == _dwGlobalTotal) ) { // check if this the none group. Only non-DCs will return this group. nasStatus = NetGroupGetInfo( szServer, pGroupUserInfo->grui0_name, 2, (LPBYTE *) &pGroupInfo2 ); hr = HRESULT_FROM_WIN32(nasStatus); BAIL_ON_FAILURE(hr); if (pGroupInfo2->grpi2_group_id == DOMAIN_GROUP_RID_USERS) { // // Set the global total to zero - destructor will dealloc. // _dwGlobalTotal = 0; } } } error: if(pGroupInfo2) NetApiBufferFree(pGroupInfo2); RRETURN(hr); } HRESULT CWinNTUserGroupsCollectionEnum::DoLocalEnumeration() { HRESULT hr; DWORD dwStatus; DWORD dwRead = 0; WCHAR szServer[MAX_PATH]; hr = MakeUncName(_ServerName, szServer); BAIL_ON_FAILURE(hr); dwStatus = NetUserGetLocalGroups(szServer, _UserName, 0, 0, (LPBYTE*)&_pLocalBuffer, MAX_PREFERRED_LENGTH, &dwRead, &_dwLocalTotal); hr = HRESULT_FROM_WIN32(dwStatus); BAIL_ON_FAILURE(hr); if (dwRead != _dwLocalTotal) hr = E_FAIL; error: RRETURN(hr); } //+--------------------------------------------------------------------------- // // Function: CWinNTUserGroupsCollectionEnum::Next // // Synopsis: Returns cElements number of requested NetOle objects in the // array supplied in pvar. // // Arguments: [cElements] -- The number of elements requested by client // [pvar] -- ptr to array of VARIANTs to for return objects // [pcElementFetched] -- if non-NULL, then number of elements // -- actually returned is placed here // // Returns: HRESULT -- S_OK if number of elements requested are returned // -- S_FALSE if number of elements is < requested // // Modifies: // // History: 11-3-95 krishnag Created. // //---------------------------------------------------------------------------- STDMETHODIMP CWinNTUserGroupsCollectionEnum::Next( ULONG cElements, VARIANT FAR* pvar, ULONG FAR* pcElementFetched ) { ULONG cElementFetched = 0; HRESULT hr = S_OK; if (NULL == pvar) { // // Returning S_FALSE to indicate that we aren't returning // as many elements as requested. // hr = S_FALSE; } else { hr = EnumUserGroups( cElements, pvar, &cElementFetched ); } if (pcElementFetched) { *pcElementFetched = cElementFetched; } RRETURN_EXP_IF_ERR(hr); } STDMETHODIMP CWinNTUserGroupsCollectionEnum::Skip(ULONG cElements) { // // Note: we better not wrap around when we add! // _dwCurrent += cElements; _dwCurrent = min(_dwTotal, _dwCurrent); if (_dwCurrent < _dwTotal) return S_OK; return S_FALSE; } STDMETHODIMP CWinNTUserGroupsCollectionEnum::Reset() { _dwCurrent = 0; return S_OK; } HRESULT CWinNTUserGroupsCollectionEnum::EnumUserGroups( ULONG cElements, VARIANT FAR* pvar, ULONG FAR* pcElementFetched ) { HRESULT hr = S_FALSE; IDispatch *pDispatch = NULL; DWORD i = 0; while (i < cElements) { hr = GetNextUserGroup(&pDispatch); if (hr == S_FALSE) { break; } VariantInit(&pvar[i]); pvar[i].vt = VT_DISPATCH; pvar[i].pdispVal = pDispatch; (*pcElementFetched)++; i++; } RRETURN(hr); } HRESULT CWinNTUserGroupsCollectionEnum::GetNextUserGroup( IDispatch ** ppDispatch ) { HRESULT hr; LPGROUP_USERS_INFO_0 lpInfo; LPWSTR GroupName; ULONG GroupType; if (_dwCurrent >= _dwTotal) return S_FALSE; if (_dwCurrent < _dwGlobalTotal) { lpInfo = (LPGROUP_USERS_INFO_0)_pGlobalBuffer + _dwCurrent; GroupType = WINNT_GROUP_GLOBAL; } else { lpInfo = (LPGROUP_USERS_INFO_0)_pLocalBuffer + _dwCurrent - _dwGlobalTotal; GroupType = WINNT_GROUP_LOCAL; } _dwCurrent++; GroupName = lpInfo->grui0_name; // // On an error, should we try to keep going? // if (GroupType == WINNT_GROUP_GLOBAL) { hr = CWinNTGroup::CreateGroup(_ParentADsPath, _ParentType, _DomainName, _ServerName, GroupName, GroupType, ADS_OBJECT_BOUND, IID_IDispatch, _Credentials, (void **)ppDispatch ); } else { hr = CWinNTGroup::CreateGroup(_ParentADsPath, _ParentType, _DomainName, _ServerName, GroupName, GroupType, ADS_OBJECT_BOUND, IID_IDispatch, _Credentials, (void **)ppDispatch ); } if (FAILED(hr)) return S_FALSE; return S_OK; }