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

1865 lines
43 KiB
C++

//---------------------------------------------------------------------------
//
// Microsoft Windows
// Copyright (C) Microsoft Corporation, 1992 - 1995
//
// File: cuser.cxx
//
// Contents: Host user object code
//
// History: 11-1-95 krishnag Created.
//
//----------------------------------------------------------------------------
#include "winnt.hxx"
#pragma hdrstop
// Class CWinNTUser
DEFINE_IDispatch_ExtMgr_Implementation(CWinNTUser)
DEFINE_IADsExtension_ExtMgr_Implementation(CWinNTUser)
DEFINE_IADs_TempImplementation(CWinNTUser)
DEFINE_IADs_PutGetImplementation(CWinNTUser,UserClass,gdwUserTableSize)
DEFINE_IADsPropertyList_Implementation(CWinNTUser,UserClass,gdwUserTableSize)
CWinNTUser::CWinNTUser():
_pDispMgr(NULL),
_pExtMgr(NULL),
_pPropertyCache(NULL),
_ParentType(0),
_DomainName(NULL),
_ServerName(NULL),
_fPasswordSet(FALSE),
_pCCredentialsPwdHolder(NULL),
_fUseCacheForAcctLocked(TRUE),
_fComputerAcct(FALSE)
{
ENLIST_TRACKING(CWinNTUser);
}
HRESULT
CWinNTUser::CreateUser(
BSTR Parent,
ULONG ParentType,
BSTR DomainName,
BSTR ServerName,
BSTR UserName,
DWORD dwObjectState,
DWORD *pdwUserFlags, // OPTIONAL
LPWSTR szFullName, // OPTIONAL
LPWSTR szDescription, // OPTIONAL
PSID pSid, // OPTIONAL
REFIID riid,
CWinNTCredentials& Credentials,
void **ppvObj
)
{
CWinNTUser FAR * pUser = NULL;
HRESULT hr = S_OK;
hr = AllocateUserObject(&pUser);
BAIL_ON_FAILURE(hr);
ADsAssert(pUser->_pDispMgr);
hr = pUser->InitializeCoreObject(
Parent,
UserName,
USER_CLASS_NAME,
USER_SCHEMA_NAME,
CLSID_WinNTUser,
dwObjectState
);
BAIL_ON_FAILURE(hr);
pUser->_Credentials = Credentials;
//
// The server name will be NULL only when we create a user
// by SID - WinNT://S-1-321-231-231. In this case we should
// not ref the server. Parent type is used as an extra check.
//
if (!((ParentType == WINNT_COMPUTER_ID)
&& !ServerName)) {
hr = pUser->_Credentials.Ref(ServerName, DomainName, ParentType);
if (hr == HRESULT_FROM_WIN32(ERROR_BAD_ARGUMENTS)) {
//
// We had a rebind error.
// This will happen in the case where the credentials
// ref the current server which is a bdc, the users is
// a member of a global group we are going through and
// we end up trying to ref the PDC when we already have
// a connection to this comp.
hr = S_OK;
}
}
BAIL_ON_FAILURE(hr);
hr = ADsAllocString(DomainName, &pUser->_DomainName);
BAIL_ON_FAILURE(hr);
if (ParentType == WINNT_DOMAIN_ID)
{
pUser->_ParentType = WINNT_DOMAIN_ID;
ADsAssert(DomainName && DomainName[0]!=L'\0');
}
else
{
pUser->_ParentType = WINNT_COMPUTER_ID;
hr = ADsAllocString(ServerName, &pUser->_ServerName);
BAIL_ON_FAILURE(hr);
}
//
// Load ext mgr and extensions
//
hr = ADSILoadExtensionManager(
USER_CLASS_NAME,
(IADsUser *) pUser,
pUser->_pDispMgr,
Credentials,
&pUser->_pExtMgr
);
BAIL_ON_FAILURE(hr);
ADsAssert(pUser->_pExtMgr);
//
// Prepopulate the object
//
hr = pUser->Prepopulate(TRUE,
pdwUserFlags,
szFullName,
szDescription,
pSid);
BAIL_ON_FAILURE(hr);
// check if the call is from UMI
if(Credentials.GetFlags() & ADS_AUTH_RESERVED) {
//
// we do not pass riid to InitUmiObject below. This is because UMI object
// does not support IDispatch. There are several places in ADSI code where
// riid passed into this function is defaulted to IID_IDispatch -
// IADsContainer::Create for example. To handle these cases, we always
// request IID_IUnknown from the UMI object. Subsequent code within UMI
// will QI for the appropriate interface.
//
if(3 == pUser->_dwNumComponents) {
pUser->_CompClasses[0] = L"Domain";
pUser->_CompClasses[1] = L"Computer";
pUser->_CompClasses[2] = L"User";
}
else if(2 == pUser->_dwNumComponents) {
if(NULL == DomainName) {
pUser->_CompClasses[0] = L"Computer";
pUser->_CompClasses[1] = L"User";
}
else if(NULL == ServerName) {
pUser->_CompClasses[0] = L"Domain";
pUser->_CompClasses[1] = L"User";
}
else
BAIL_ON_FAILURE(hr = UMI_E_FAIL);
}
else
BAIL_ON_FAILURE(hr = UMI_E_FAIL);
hr = pUser->InitUmiObject(
pUser->_Credentials,
UserClass,
gdwUserTableSize,
pUser->_pPropertyCache,
(IUnknown *) (INonDelegatingUnknown *) pUser,
pUser->_pExtMgr,
IID_IUnknown,
ppvObj
);
BAIL_ON_FAILURE(hr);
//
// UMI object was created and the interface was obtained successfully.
// UMI object now has a reference to the inner unknown of IADs, since
// the call to Release() below is not going to be made in this case.
//
RRETURN(hr);
}
hr = pUser->QueryInterface(riid, ppvObj);
BAIL_ON_FAILURE(hr);
pUser->Release();
RRETURN(hr);
error:
delete pUser;
RRETURN_EXP_IF_ERR(hr);
}
HRESULT
CWinNTUser::CreateUser(
BSTR Parent,
ULONG ParentType,
BSTR DomainName,
BSTR ServerName,
BSTR UserName,
DWORD dwObjectState,
REFIID riid,
CWinNTCredentials& Credentials,
void **ppvObj
)
{
HRESULT hr = S_OK;
hr = CWinNTUser::CreateUser(
Parent,
ParentType,
DomainName,
ServerName,
UserName,
dwObjectState,
NULL,
NULL,
NULL,
NULL,
riid,
Credentials,
ppvObj
);
RRETURN_EXP_IF_ERR(hr);
}
CWinNTUser::~CWinNTUser( )
{
ADsFreeString(_DomainName);
ADsFreeString(_ServerName);
delete _pExtMgr; // created last, destroyed first
delete _pDispMgr;
delete _pPropertyCache;
if (_pCCredentialsPwdHolder) {
delete _pCCredentialsPwdHolder;
}
}
//----------------------------------------------------------------------------
// Function: QueryInterface
//
// Synopsis: If this object is aggregated within another object, then
// all calls will delegate to the outer object. Otherwise, the
// non-delegating QI is called
//
// Arguments:
//
// iid interface requested
// ppInterface Returns pointer to interface requested. NULL if interface
// is not supported.
//
// Returns: S_OK on success. Error code otherwise.
//
// Modifies: *ppInterface to return interface pointer
//
//----------------------------------------------------------------------------
STDMETHODIMP CWinNTUser::QueryInterface(
REFIID iid,
LPVOID *ppInterface
)
{
if(_pUnkOuter != NULL)
RRETURN(_pUnkOuter->QueryInterface(
iid,
ppInterface
));
RRETURN(NonDelegatingQueryInterface(
iid,
ppInterface
));
}
//----------------------------------------------------------------------------
// Function: AddRef
//
// Synopsis: IUnknown::AddRef. If this object is aggregated within
// another, all calls will delegate to the outer object.
// Otherwise, the non-delegating AddRef is called
//
// Arguments:
//
// None
//
// Returns: New reference count
//
// Modifies: Nothing
//
//----------------------------------------------------------------------------
STDMETHODIMP_(ULONG) CWinNTUser::AddRef(void)
{
if(_pUnkOuter != NULL)
RRETURN(_pUnkOuter->AddRef());
RRETURN(NonDelegatingAddRef());
}
//----------------------------------------------------------------------------
// Function: Release
//
// Synopsis: IUnknown::Release. If this object is aggregated within
// another, all calls will delegate to the outer object.
// Otherwise, the non-delegating Release is called
//
// Arguments:
//
// None
//
// Returns: New reference count
//
// Modifies: Nothing
//
//----------------------------------------------------------------------------
STDMETHODIMP_(ULONG) CWinNTUser::Release(void)
{
if(_pUnkOuter != NULL)
RRETURN(_pUnkOuter->Release());
RRETURN(NonDelegatingRelease());
}
//----------------------------------------------------------------------------
STDMETHODIMP
CWinNTUser::NonDelegatingQueryInterface(
REFIID iid,
LPVOID FAR* ppv
)
{
HRESULT hr = S_OK;
if (ppv == NULL) {
RRETURN(E_POINTER);
}
if (IsEqualIID(iid, IID_IUnknown))
{
*ppv = (IADsUser FAR *) this;
}
else if (IsEqualIID(iid, IID_IADsUser))
{
*ppv = (IADsUser FAR *) this;
}
else if (IsEqualIID(iid, IID_IADs))
{
*ppv = (IADsUser FAR *) this;
}
else if (IsEqualIID(iid, IID_IADsPropertyList))
{
*ppv = (IADsPropertyList FAR *) this;
}
else if (IsEqualIID(iid, IID_IDispatch))
{
*ppv = (IADsUser FAR *) this;
}
else if (IsEqualIID(iid, IID_ISupportErrorInfo))
{
*ppv = (ISupportErrorInfo FAR *) this;
}
else if( (_pDispatch != NULL) &&
IsEqualIID(iid, IID_IADsExtension) )
{
*ppv = (IADsExtension *) this;
}
else if (_pExtMgr)
{
RRETURN(_pExtMgr->QueryInterface(iid,ppv));
}
else
{
*ppv = NULL;
return E_NOINTERFACE;
}
AddRef();
return NOERROR;
}
/* ISupportErrorInfo method */
STDMETHODIMP
CWinNTUser::InterfaceSupportsErrorInfo(
THIS_ REFIID riid
)
{
if (IsEqualIID(riid, IID_IADs) ||
IsEqualIID(riid, IID_IADsPropertyList) ||
IsEqualIID(riid, IID_IADsUser)) {
RRETURN(S_OK);
} else {
RRETURN(S_FALSE);
}
}
/* IADs methods */
STDMETHODIMP
CWinNTUser::SetInfo(THIS)
{
HRESULT hr = S_OK;
NET_API_STATUS nasStatus;
WCHAR szPDCName[MAX_PATH];
WCHAR *pszPassword = NULL;
WCHAR *pszServerName = _ServerName;
//
// objects associated with invalid SIDs have neither a
// corresponding server nor domain
//
if ((!_DomainName) && (!_ServerName)) {
BAIL_ON_FAILURE(hr = E_ADS_INVALID_USER_OBJECT);
}
if (GetObjectState() == ADS_OBJECT_UNBOUND) {
if (_ParentType == WINNT_DOMAIN_ID) {
hr = WinNTGetCachedDCName(
_DomainName,
szPDCName,
_Credentials.GetFlags()
);
BAIL_ON_FAILURE(hr);
//
// + 2 skips the backslashes when calling create
//
pszServerName = szPDCName + 2;
}
if (!_fPasswordSet) {
hr = WinNTCreateUser(
pszServerName,
_Name
);
} else {
hr = getPrivatePassword(&pszPassword);
BAIL_ON_FAILURE(hr);
hr = WinNTCreateUser(
pszServerName,
_Name,
pszPassword
);
}
BAIL_ON_FAILURE(hr);
SetObjectState(ADS_OBJECT_BOUND);
}
hr = SetInfo(3);
if(SUCCEEDED(hr))
_pPropertyCache->ClearModifiedFlags();
error:
if (pszPassword) {
SecureZeroMemory(pszPassword, sizeof(WCHAR) * wcslen(pszPassword));
FreeADsStr(pszPassword);
}
RRETURN_EXP_IF_ERR(hr);
}
STDMETHODIMP
CWinNTUser::GetInfo(THIS)
{
HRESULT hr;
_pPropertyCache->flushpropcache();
// USER_INFO
//
hr = GetInfo(
3,
TRUE
);
BAIL_ON_FAILURE(hr);
//
// USER_MODAL_INFO0
//
hr = GetInfo(
10,
TRUE
);
BAIL_ON_FAILURE(hr);
//
// USER_MODAL_INFO3
//
hr = GetInfo(
13,
TRUE
);
BAIL_ON_FAILURE(hr);
//
// objectSid. LookupAccountName fails for machine accounts on NT4, but
// works on Win2K. In order for an explicit GetInfo to succeed against NT4
// systems we do not check the error code returned below. If this call
// fails, a subsequent Get("ObjectSid") will return
// E_ADS_PROPERTY_NOT_FOUND.
//
GetInfo(
20,
TRUE
);
error :
RRETURN(hr);
}
STDMETHODIMP
CWinNTUser::ImplicitGetInfo(THIS)
{
HRESULT hr;
// USER_INFO
//
hr = GetInfo(
3,
FALSE
);
BAIL_ON_FAILURE(hr);
//
// USER_MODAL_INFO0
//
hr = GetInfo(
10,
FALSE
);
BAIL_ON_FAILURE(hr);
//
// USER_MODAL_INFO3
//
hr = GetInfo(
13,
FALSE
);
BAIL_ON_FAILURE(hr);
//
// objectSid
//
hr = GetInfo(
20,
FALSE
);
error :
RRETURN(hr);
}
HRESULT
CWinNTUser::AllocateUserObject(
CWinNTUser ** ppUser
)
{
CWinNTUser FAR * pUser = NULL;
CAggregatorDispMgr FAR * pDispMgr = NULL;
CPropertyCache FAR * pPropertyCache = NULL;
HRESULT hr = S_OK;
pUser = new CWinNTUser();
if (pUser == NULL) {
hr = E_OUTOFMEMORY;
}
BAIL_ON_FAILURE(hr);
pDispMgr = new CAggregatorDispMgr;
if (pDispMgr == NULL) {
hr = E_OUTOFMEMORY;
}
BAIL_ON_FAILURE(hr);
hr = LoadTypeInfoEntry(
pDispMgr,
LIBID_ADs,
IID_IADsUser,
(IADsUser *)pUser,
DISPID_REGULAR
);
BAIL_ON_FAILURE(hr);
hr = LoadTypeInfoEntry(
pDispMgr,
LIBID_ADs,
IID_IADsPropertyList,
(IADsPropertyList *)pUser,
DISPID_VALUE
);
BAIL_ON_FAILURE(hr);
hr = CPropertyCache::createpropertycache(
UserClass,
gdwUserTableSize,
(CCoreADsObject *)pUser,
&pPropertyCache
);
BAIL_ON_FAILURE(hr);
pDispMgr->RegisterPropertyCache(
pPropertyCache
);
pUser->_pPropertyCache = pPropertyCache;
pUser->_pDispMgr = pDispMgr;
*ppUser = pUser;
RRETURN(hr);
error:
delete pDispMgr;
delete pPropertyCache;
delete pUser;
RRETURN(hr);
}
//
// For current implementation in clocgroup:
// If this function is called as a public function (ie. by another
// modual/class), fExplicit must be FALSE since the cache is NOT
// flushed in this function.
//
// External functions should ONLY call GetInfo(no param) for explicit
// GetInfo. This will flush the cache properly.
//
STDMETHODIMP
CWinNTUser::GetInfo(
THIS_ DWORD dwApiLevel,
BOOL fExplicit
)
{
HRESULT hr = S_OK;
switch (dwApiLevel) {
// GetInfo(1 or 2, fExplicit) in ADSI codes should be modified
// to GetInfo(3, fExplicit) to minimize calls on wire.
case 3:
hr = GetStandardInfo(3, fExplicit);
RRETURN_EXP_IF_ERR(hr);
case 10:
hr = GetModalInfo(0, fExplicit);
RRETURN_EXP_IF_ERR(hr);
case 13:
hr = GetModalInfo(3, fExplicit);
RRETURN_EXP_IF_ERR(hr);
case 20:
hr = GetSidInfo(fExplicit);
RRETURN_EXP_IF_ERR(hr);
default:
RRETURN_EXP_IF_ERR(E_ADS_BAD_PARAMETER);
}
}
HRESULT
CWinNTUser::GetStandardInfo(
THIS_ DWORD dwApiLevel,
BOOL fExplicit
)
{
NET_API_STATUS nasStatus;
LPBYTE lpBuffer = NULL;
HRESULT hr;
WCHAR szHostServerName[MAX_PATH];
if (GetObjectState() == ADS_OBJECT_UNBOUND) {
RRETURN_EXP_IF_ERR(E_ADS_OBJECT_UNBOUND);
}
//
// objects associated with invalid SIDs have neither a
// corresponding server nor domain
//
if ((!_DomainName) && (!_ServerName)) {
BAIL_ON_FAILURE(hr = E_ADS_INVALID_USER_OBJECT);
}
if (_ParentType == WINNT_DOMAIN_ID) {
hr = WinNTGetCachedDCName(
_DomainName,
szHostServerName,
_Credentials.GetFlags()
);
BAIL_ON_FAILURE(hr);
}else {
hr = MakeUncName(
_ServerName,
szHostServerName
);
BAIL_ON_FAILURE(hr);
}
nasStatus = NetUserGetInfo(
szHostServerName,
_Name,
dwApiLevel,
&lpBuffer
);
hr = HRESULT_FROM_WIN32(nasStatus);
BAIL_ON_FAILURE(hr);
hr = UnMarshall(
lpBuffer,
dwApiLevel,
fExplicit
);
BAIL_ON_FAILURE(hr);
error:
if (lpBuffer) {
NetApiBufferFree(lpBuffer);
}
RRETURN_EXP_IF_ERR(hr);
}
HRESULT
CWinNTUser::UnMarshall(
LPBYTE lpBuffer,
DWORD dwApiLevel,
BOOL fExplicit
)
{
VARIANT_BOOL fBool;
BSTR bstrData = NULL;
LONG lnData = 0L;
VARIANT vaData;
DATE daDate = 0;
ADsAssert(lpBuffer);
switch (dwApiLevel) {
// GetStandardInfo currently only be called with dwApiLevel=3. If
// dwApiLevel = 1 or 2 is used, modify ADSI codes to 3.
case 3:
RRETURN(UnMarshall_Level3(fExplicit, (LPUSER_INFO_3)lpBuffer));
break;
default:
RRETURN(E_FAIL);
}
}
HRESULT
CWinNTUser::UnMarshall_Level3(
BOOL fExplicit,
LPUSER_INFO_3 pUserInfo3
)
{
HRESULT hr = S_OK;
//
// Begin Account Restrictions Properties
//
hr = SetDWORDPropertyInCache(
_pPropertyCache,
TEXT("UserFlags"),
pUserInfo3->usri3_flags,
fExplicit
);
if(SUCCEEDED(hr)) {
_fUseCacheForAcctLocked = TRUE;
}
if( (pUserInfo3->usri3_flags & UF_WORKSTATION_TRUST_ACCOUNT) ||
(pUserInfo3->usri3_flags & UF_SERVER_TRUST_ACCOUNT) ||
(pUserInfo3->usri3_flags & UF_INTERDOMAIN_TRUST_ACCOUNT) ) {
_fComputerAcct = TRUE;
}
//
// If usri3_acct_expires == TIMEQ_FOREVER, it means we need
// to ignore the acct expiration date, the account
// can never expire.
//
if (pUserInfo3->usri3_acct_expires != TIMEQ_FOREVER) {
hr = SetDATE70PropertyInCache(
_pPropertyCache,
TEXT("AccountExpirationDate"),
pUserInfo3->usri3_acct_expires,
fExplicit
);
}
hr = SetDelimitedStringPropertyInCache(
_pPropertyCache,
TEXT("LoginWorkstations"),
pUserInfo3->usri3_workstations,
fExplicit
);
hr = SetDWORDPropertyInCache(
_pPropertyCache,
TEXT("MaxStorage"),
pUserInfo3->usri3_max_storage,
fExplicit
);
hr = SetDWORDPropertyInCache(
_pPropertyCache,
TEXT("PasswordAge"),
pUserInfo3->usri3_password_age,
fExplicit
);
hr = SetDWORDPropertyInCache(
_pPropertyCache,
TEXT("PasswordExpired"),
pUserInfo3->usri3_password_expired,
fExplicit
);
hr = SetOctetPropertyInCache(
_pPropertyCache,
TEXT("LoginHours"),
pUserInfo3->usri3_logon_hours,
21,
fExplicit
);
//
// Begin Business Info Properties
//
hr = SetLPTSTRPropertyInCache(
_pPropertyCache,
TEXT("FullName"),
pUserInfo3->usri3_full_name,
fExplicit
);
hr = SetLPTSTRPropertyInCache(
_pPropertyCache,
TEXT("Description"),
pUserInfo3->usri3_comment,
fExplicit
);
//
// Begin Account Statistics Properties
//
hr = SetDWORDPropertyInCache(
_pPropertyCache,
TEXT("BadPasswordAttempts"),
pUserInfo3->usri3_bad_pw_count,
fExplicit
);
//
// lasg_logon/off == 0 means user never logon/off or logon/off time unknown.
//
if (pUserInfo3->usri3_last_logon!=0) {
hr = SetDATE70PropertyInCache(
_pPropertyCache,
TEXT("LastLogin"),
pUserInfo3->usri3_last_logon,
fExplicit
);
}
if (pUserInfo3->usri3_last_logoff!=0) {
hr = SetDATE70PropertyInCache(
_pPropertyCache,
TEXT("LastLogoff"),
pUserInfo3->usri3_last_logoff,
fExplicit
);
}
//
// Begin Other Info Properties
//
hr = SetLPTSTRPropertyInCache(
_pPropertyCache,
TEXT("HomeDirectory"),
pUserInfo3->usri3_home_dir,
fExplicit
);
hr = SetLPTSTRPropertyInCache(
_pPropertyCache,
TEXT("LoginScript"),
pUserInfo3->usri3_script_path,
fExplicit
);
hr = SetLPTSTRPropertyInCache(
_pPropertyCache,
TEXT("Profile"),
pUserInfo3->usri3_profile,
fExplicit
);
hr = SetLPTSTRPropertyInCache(
_pPropertyCache,
TEXT("HomeDirDrive"),
pUserInfo3->usri3_home_dir_drive,
fExplicit
);
hr = SetLPTSTRPropertyInCache(
_pPropertyCache,
TEXT("Parameters"),
pUserInfo3->usri3_parms,
fExplicit
);
hr = SetDWORDPropertyInCache(
_pPropertyCache,
TEXT("PrimaryGroupID"),
pUserInfo3->usri3_primary_group_id,
fExplicit
);
hr = SetLPTSTRPropertyInCache(
_pPropertyCache,
TEXT("Name"),
_Name,
fExplicit
);
RRETURN(S_OK);
}
HRESULT
CWinNTUser::Prepopulate(
BOOL fExplicit,
DWORD *pdwUserFlags, // OPTIONAL
LPWSTR szFullName, // OPTIONAL
LPWSTR szDescription, // OPTIONAL
PSID pSid // OPTIONAL
)
{
HRESULT hr = S_OK;
DWORD dwErr = 0;
DWORD dwSidLength = 0;
//
// Prepopulate the object with supplied info,
// if available
//
if (pdwUserFlags) {
hr = SetDWORDPropertyInCache(
_pPropertyCache,
TEXT("UserFlags"),
*pdwUserFlags,
TRUE
);
BAIL_ON_FAILURE(hr);
//
// see comment on _fUseCacheForAcctLocked in cuser.hxx
//
_fUseCacheForAcctLocked = FALSE;
}
if (szFullName) {
hr = SetLPTSTRPropertyInCache(
_pPropertyCache,
TEXT("FullName"),
szFullName,
TRUE
);
BAIL_ON_FAILURE(hr);
}
if (szDescription) {
hr = SetLPTSTRPropertyInCache(
_pPropertyCache,
TEXT("Description"),
szDescription,
TRUE
);
BAIL_ON_FAILURE(hr);
}
if (pSid) {
//
// On NT4 for some reason GetLengthSID does not set lasterror to 0
//
SetLastError(NO_ERROR);
dwSidLength = GetLengthSid(pSid);
//
// This is an extra check to make sure that we have the
// correct length.
//
dwErr = GetLastError();
if (dwErr != NO_ERROR) {
hr = HRESULT_FROM_WIN32(dwErr);
BAIL_ON_FAILURE(hr);
}
hr = SetOctetPropertyInCache(
_pPropertyCache,
TEXT("objectSid"),
(PBYTE) pSid,
dwSidLength,
TRUE
);
BAIL_ON_FAILURE(hr);
}
error:
RRETURN(hr);
}
HRESULT
CWinNTUser::GetSidInfo(
IN BOOL fExplicit
)
{
HRESULT hr = E_FAIL;
WCHAR szHostServerName[MAX_PATH];
//
// objects associated with invalid SIDs have neither a
// corresponding server nor domain
//
if ((!_DomainName) && (!_ServerName)) {
BAIL_ON_FAILURE(hr = E_ADS_INVALID_USER_OBJECT);
}
//
// Get Server Name
//
if (_ParentType == WINNT_DOMAIN_ID) {
hr = WinNTGetCachedDCName(
_DomainName,
szHostServerName,
_Credentials.GetFlags()
);
BAIL_ON_FAILURE(hr);
}else {
hr = MakeUncName(
_ServerName,
szHostServerName
);
BAIL_ON_FAILURE(hr);
}
//
// Get Sid of this user account and store in cache if fExplicit.
//
hr = GetSidIntoCache(
szHostServerName,
_Name,
_pPropertyCache,
fExplicit
);
BAIL_ON_FAILURE(hr);
error:
RRETURN(hr);
}
HRESULT
CWinNTUser::SetInfo(THIS_ DWORD dwApiLevel)
{
NET_API_STATUS nasStatus;
HRESULT hr;
LPBYTE lpBuffer = NULL;
DWORD dwParamErr = 0;
WCHAR szHostServerName[MAX_PATH];
//
// objects associated with invalid SIDs have neither a
// corresponding server nor domain
//
if ((!_DomainName) && (!_ServerName)) {
BAIL_ON_FAILURE(hr = E_ADS_INVALID_USER_OBJECT);
}
if (_ParentType == WINNT_DOMAIN_ID) {
hr = WinNTGetCachedDCName(
_DomainName,
szHostServerName,
_Credentials.GetFlags()
);
BAIL_ON_FAILURE(hr);
}else {
hr = MakeUncName(
_ServerName,
szHostServerName
);
BAIL_ON_FAILURE(hr);
}
nasStatus = NetUserGetInfo(
szHostServerName,
_Name,
dwApiLevel,
&lpBuffer
);
hr = HRESULT_FROM_WIN32(nasStatus);
BAIL_ON_FAILURE(hr);
hr = MarshallAndSet(szHostServerName, lpBuffer, dwApiLevel);
BAIL_ON_FAILURE(hr);
error:
if (lpBuffer) {
NetApiBufferFree(lpBuffer);
}
RRETURN_EXP_IF_ERR(hr);
}
HRESULT
CWinNTUser::MarshallAndSet(
LPWSTR szHostServerName,
LPBYTE lpBuffer,
DWORD dwApiLevel
)
{
ADsAssert(lpBuffer);
switch (dwApiLevel) {
//
// dwApiLevel = 1 or 2 should change to 3 in caller codes to min
// calls on wire
case 3:
RRETURN(Marshall_Set_Level3(szHostServerName, (LPUSER_INFO_3)lpBuffer));
break;
//
// caae 10:
// case 13:
// USER_MODAL_INFO should be set at domain level,
// Not at user level
//
//
// case 20:
// objectSid not writable
//
default:
RRETURN(E_FAIL);
}
}
HRESULT
CWinNTUser::Marshall_Set_Level3(
LPWSTR szHostServerName,
LPUSER_INFO_3 pUserInfo3
)
{
HRESULT hr;
DWORD dwFlags = 0;
DWORD dwAcctExpDate = 0;
LPWSTR pszDescription = NULL;
LPWSTR pszFullName = NULL;
DWORD dwBadPwCount = 0;
DWORD dwLastLogin = 0;
DWORD dwLastLogoff = 0;
LPWSTR pszHomeDir = NULL;
LPWSTR pszScript = NULL;
LPWSTR pszProfile = NULL;
LPWSTR pszLoginWorkstations = NULL;
DWORD dwMaxStorage = 0;
LPWSTR pszHomeDirDrive = NULL;
LPWSTR pszParameters = NULL;
DWORD dwPrimaryGroupId = 0;
DWORD dwPasswordExpired = 0;
OctetString octetString;
DWORD dwParmErr = 0;
NET_API_STATUS nasStatus;
hr = GetDWORDPropertyFromCache(
_pPropertyCache,
TEXT("UserFlags"),
&dwFlags
);
if(SUCCEEDED(hr)){
pUserInfo3->usri3_flags = dwFlags;
}
hr = GetDATE70PropertyFromCache(
_pPropertyCache,
TEXT("AccountExpirationDate"),
&dwAcctExpDate
);
if(SUCCEEDED(hr)){
//
// Pick an easy to remeber date to represent "account never expires" :
// 1/1/70 at 0:00. (Range <= 86400 and >= 0xffffffff-86400 is +/- one
// day from 1/1/70 at 0:00 to take time localization into account.)
//
if (dwAcctExpDate <= 86400 || dwAcctExpDate >= (0xffffffff-86400)) {
pUserInfo3->usri3_acct_expires = TIMEQ_FOREVER;
}
else {
pUserInfo3->usri3_acct_expires = dwAcctExpDate;
}
}
hr = GetDWORDPropertyFromCache(
_pPropertyCache,
TEXT("PasswordExpired"),
&dwPasswordExpired
);
if(SUCCEEDED(hr)){
pUserInfo3->usri3_password_expired = dwPasswordExpired;
}
hr = GetDWORDPropertyFromCache(
_pPropertyCache,
TEXT("MaxStorage"),
&dwMaxStorage
);
if(SUCCEEDED(hr)){
pUserInfo3->usri3_max_storage = dwMaxStorage;
}
hr = GetDelimitedStringPropertyFromCache(
_pPropertyCache,
TEXT("LoginWorkstations"),
&pszLoginWorkstations
);
if(SUCCEEDED(hr)){
pUserInfo3->usri3_workstations = pszLoginWorkstations;
}
//
// Begin Business Information Properties
//
hr = GetLPTSTRPropertyFromCache(
_pPropertyCache,
TEXT("Description"),
&pszDescription
);
if(SUCCEEDED(hr)){
pUserInfo3->usri3_comment = pszDescription;
}
hr = GetLPTSTRPropertyFromCache(
_pPropertyCache,
TEXT("FullName"),
&pszFullName
);
if(SUCCEEDED(hr)){
pUserInfo3->usri3_full_name = pszFullName;
}
hr = GetOctetPropertyFromCache(
_pPropertyCache,
TEXT("LoginHours"),
&octetString
);
if(SUCCEEDED(hr)){
//
// We can only assume what the size of the usri3_logon_hours
// buffer is, but according to MSDN, it should be 21 bytes.
// Make sure that no more than 21 bytes is copied.
//
memcpy(pUserInfo3->usri3_logon_hours,
octetString.pByte,
(octetString.dwSize>21)?21:octetString.dwSize);
FreeADsMem(octetString.pByte);
}
/*
//
// Begin Account Statistics Properties - should not be writable.
//
hr = GetDWORDPropertyFromCache(
_pPropertyCache,
TEXT("BadPasswordAttempts"),
&dwBadPwCount
);
if(SUCCEEDED(hr)){
pUserInfo3->usri3_bad_pw_count = dwBadPwCount;
}
hr = GetDATE70PropertyFromCache(
_pPropertyCache,
TEXT("LastLogin"),
&dwLastLogin
);
if(SUCCEEDED(hr)){
pUserInfo3->usri3_last_logon = dwLastLogin;
}
hr = GetDATE70PropertyFromCache(
_pPropertyCache,
TEXT("LastLogoff"),
&dwLastLogoff
);
if(SUCCEEDED(hr)){
pUserInfo3->usri3_last_logoff = dwLastLogoff;
}
*/
//
// Begin Other Info Properties
//
hr = GetLPTSTRPropertyFromCache(
_pPropertyCache,
TEXT("HomeDirectory"),
&pszHomeDir
);
if(SUCCEEDED(hr)){
pUserInfo3->usri3_home_dir = pszHomeDir;
}
hr = GetLPTSTRPropertyFromCache(
_pPropertyCache,
TEXT("LoginScript"),
&pszScript
);
if(SUCCEEDED(hr)){
pUserInfo3->usri3_script_path = pszScript;
}
hr = GetLPTSTRPropertyFromCache(
_pPropertyCache,
TEXT("Profile"),
&pszProfile
);
if(SUCCEEDED(hr)){
pUserInfo3->usri3_profile = pszProfile;
}
hr = GetLPTSTRPropertyFromCache(
_pPropertyCache,
TEXT("HomeDirDrive"),
&pszHomeDirDrive
);
if(SUCCEEDED(hr)){
pUserInfo3->usri3_home_dir_drive = pszHomeDirDrive;
}
hr = GetLPTSTRPropertyFromCache(
_pPropertyCache,
TEXT("Parameters"),
&pszParameters
);
if(SUCCEEDED(hr)){
pUserInfo3->usri3_parms = pszParameters;
}
hr = GetDWORDPropertyFromCache(
_pPropertyCache,
TEXT("PrimaryGroupID"),
&dwPrimaryGroupId
);
if(SUCCEEDED(hr)){
pUserInfo3->usri3_primary_group_id = dwPrimaryGroupId;
}
//
// Now perform the Set call.
//
nasStatus = NetUserSetInfo(
szHostServerName,
_Name,
3,
(LPBYTE)pUserInfo3,
&dwParmErr
);
hr = HRESULT_FROM_WIN32(nasStatus);
BAIL_ON_FAILURE(hr);
error:
if (pszDescription) {
FreeADsStr(pszDescription);
}
if (pszFullName) {
FreeADsStr(pszFullName);
}
if (pszHomeDir) {
FreeADsStr(pszHomeDir);
}
if (pszScript) {
FreeADsStr(pszScript);
}
if (pszProfile) {
FreeADsStr(pszProfile);
}
if (pszLoginWorkstations) {
FreeADsStr(pszLoginWorkstations);
}
if (pszParameters) {
FreeADsStr(pszParameters);
}
if (pszHomeDirDrive) {
FreeADsStr(pszHomeDirDrive);
}
RRETURN(hr);
}
HRESULT
CWinNTUser::Marshall_Create_Level1(
LPWSTR szHostServerName,
LPUSER_INFO_1 pUserInfo1
)
{
HRESULT hr = S_OK;
NET_API_STATUS nasStatus;
DWORD dwParmErr;
pUserInfo1->usri1_name = _Name;
pUserInfo1->usri1_password = NULL;
pUserInfo1->usri1_password_age = DEF_MAX_PWAGE;
pUserInfo1->usri1_priv = 1;
pUserInfo1->usri1_home_dir = NULL;
pUserInfo1->usri1_comment = NULL;
pUserInfo1->usri1_flags = 0x00000201;
pUserInfo1->usri1_script_path = NULL;
nasStatus = NetUserAdd(
szHostServerName,
1,
(LPBYTE)pUserInfo1,
&dwParmErr
);
hr = HRESULT_FROM_WIN32(nasStatus);
RRETURN(hr);
}
HRESULT
CWinNTUser::GetModalInfo(
THIS_ DWORD dwApiLevel,
BOOL fExplicit
)
{
NET_API_STATUS nasStatus;
LPBYTE lpBuffer = NULL;
HRESULT hr;
WCHAR szPDCName[MAX_PATH];
//
// objects associated with invalid SIDs have neither a
// corresponding server nor domain
//
if ((!_DomainName) && (!_ServerName)) {
BAIL_ON_FAILURE(hr = E_ADS_INVALID_USER_OBJECT);
}
if (_ParentType == WINNT_DOMAIN_ID) {
hr = WinNTGetCachedDCName(
_DomainName,
szPDCName,
_Credentials.GetFlags()
);
BAIL_ON_FAILURE(hr);
}else {
hr = MakeUncName(
_ServerName,
szPDCName
);
BAIL_ON_FAILURE(hr);
}
nasStatus = NetUserModalsGet(
szPDCName,
dwApiLevel,
&lpBuffer
);
hr = HRESULT_FROM_WIN32(nasStatus);
BAIL_ON_FAILURE(hr);
hr = UnMarshallModalInfo(lpBuffer, dwApiLevel, fExplicit);
BAIL_ON_FAILURE(hr);
error:
if (lpBuffer) {
NetApiBufferFree(lpBuffer);
}
RRETURN(hr);
}
HRESULT
CWinNTUser::UnMarshallModalInfo(
LPBYTE lpBuffer,
DWORD dwApiLevel,
BOOL fExplicit
)
{
ADsAssert(lpBuffer);
switch (dwApiLevel) {
case 0:
RRETURN(UnMarshall_ModalLevel0(fExplicit, (LPUSER_MODALS_INFO_0)lpBuffer));
break;
case 2:
RRETURN(UnMarshall_ModalLevel2(fExplicit, (LPUSER_MODALS_INFO_2)lpBuffer));
break;
case 3:
RRETURN(UnMarshall_ModalLevel3(fExplicit, (LPUSER_MODALS_INFO_3)lpBuffer));
break;
default:
RRETURN(E_FAIL);
}
}
HRESULT
CWinNTUser::UnMarshall_ModalLevel0(
BOOL fExplicit,
LPUSER_MODALS_INFO_0 pUserInfo0
)
{
HRESULT hr = S_OK;
hr = SetDWORDPropertyInCache(
_pPropertyCache,
TEXT("MinPasswordLength"),
pUserInfo0->usrmod0_min_passwd_len,
fExplicit
);
hr = SetDWORDPropertyInCache(
_pPropertyCache,
TEXT("MaxPasswordAge"),
pUserInfo0->usrmod0_max_passwd_age,
fExplicit
);
hr = SetDWORDPropertyInCache(
_pPropertyCache,
TEXT("MinPasswordAge"),
pUserInfo0->usrmod0_min_passwd_age,
fExplicit
);
hr = SetDWORDPropertyInCache(
_pPropertyCache,
TEXT("PasswordHistoryLength"),
pUserInfo0->usrmod0_password_hist_len,
fExplicit
);
RRETURN(S_OK);
}
HRESULT
CWinNTUser::UnMarshall_ModalLevel2(
BOOL fExplicit,
LPUSER_MODALS_INFO_2 pUserInfo2
)
{
RRETURN(S_OK);
}
HRESULT
CWinNTUser::UnMarshall_ModalLevel3(
BOOL fExplicit,
LPUSER_MODALS_INFO_3 pUserInfo3
)
{
HRESULT hr = S_OK;
hr = SetDWORDPropertyInCache(
_pPropertyCache,
TEXT("AutoUnlockInterval"),
pUserInfo3->usrmod3_lockout_duration,
fExplicit
);
hr = SetDWORDPropertyInCache(
_pPropertyCache,
TEXT("LockoutObservationInterval"),
pUserInfo3->usrmod3_lockout_observation_window,
fExplicit
);
hr = SetDWORDPropertyInCache(
_pPropertyCache,
TEXT("MaxBadPasswordsAllowed"),
pUserInfo3->usrmod3_lockout_threshold,
fExplicit
);
RRETURN(S_OK);
}
//
// This method is meant to set the password, so that new users
// can be created, their password set and then SetInfo can be
// called. This is necessary to allow creation of users when there
// are restrictions such as passwd should be present.
//
HRESULT
CWinNTUser::setPrivatePassword(
PWSTR pszNewPassword
)
{
HRESULT hr = S_OK;
// CCred safely stores password for us
if (_pCCredentialsPwdHolder) {
hr = _pCCredentialsPwdHolder->SetPassword(pszNewPassword);
BAIL_ON_FAILURE(hr);
} else
_pCCredentialsPwdHolder = new CCredentials(NULL, pszNewPassword, 0);
if (!_pCCredentialsPwdHolder) {
hr = E_OUTOFMEMORY;
} else
_fPasswordSet = TRUE;
error:
RRETURN(hr);
}
//
// This method is meant to set the password, so that new users
// can be created, their password set and then SetInfo can be
// called. This is necessary to allow creation of users when there
// are restrictions such as passwd should be present.
//
HRESULT
CWinNTUser::getPrivatePassword(
PWSTR * ppszPassword
)
{
HRESULT hr = S_OK;
if (_pCCredentialsPwdHolder && _fPasswordSet) {
hr = _pCCredentialsPwdHolder->GetPassword(ppszPassword);
} else
hr = E_FAIL;
RRETURN(hr);
}
HRESULT
CWinNTUser::GetUserFlags(
DWORD *pdwUserFlags
)
{
HRESULT hr = S_OK;
NET_API_STATUS nasStatus;
LPBYTE lpBuffer = NULL;
WCHAR szHostServerName[MAX_PATH];
ADsAssert(pdwUserFlags != NULL);
if (GetObjectState() == ADS_OBJECT_UNBOUND) {
RRETURN_EXP_IF_ERR(E_ADS_OBJECT_UNBOUND);
}
//
// objects associated with invalid SIDs have neither a
// corresponding server nor domain
//
if ((!_DomainName) && (!_ServerName)) {
BAIL_ON_FAILURE(hr = E_ADS_INVALID_USER_OBJECT);
}
if (_ParentType == WINNT_DOMAIN_ID) {
hr = WinNTGetCachedDCName(
_DomainName,
szHostServerName,
_Credentials.GetFlags()
);
BAIL_ON_FAILURE(hr);
}else {
hr = MakeUncName(
_ServerName,
szHostServerName
);
BAIL_ON_FAILURE(hr);
}
nasStatus = NetUserGetInfo(
szHostServerName,
_Name,
3,
&lpBuffer
);
hr = HRESULT_FROM_WIN32(nasStatus);
BAIL_ON_FAILURE(hr);
*pdwUserFlags = ((LPUSER_INFO_3)lpBuffer)->usri3_flags;
error:
if (lpBuffer) {
NetApiBufferFree(lpBuffer);
}
RRETURN(hr);
}