Изменение пароля пользователем не удается изменить (поставщик LDAP)
Возможность пользователя изменять собственный пароль — это разрешение, которое может быть предоставлено или запрещено. Чтобы запретить это разрешение, задайте два acES в списке управления доступом для дескриптора безопасности (DACL) объекта пользователя с типом ADS_ACETYPE_ACCESS_DENIED_OBJECT ace. Один ACE запрещает разрешение пользователю, а другой ACE запрещает разрешение группе "Все". Оба acES являются объектами запрета ACEs, которые указывают GUID расширенного разрешения для изменения паролей. Чтобы предоставить это разрешение, задайте те же acES с типом ADS_ACETYPE_ACCESS_ALLOWED_OBJECT ace.
В следующей процедуре описывается изменение или добавление acES для этого разрешения.
Изменение или добавление acES для этого разрешения
Привязка к объекту пользователя.
Получите объект IADsSecurityDescriptor из свойства ntSecurityDescriptor объекта пользователя.
Получите интерфейс IADsAccessControlList для дескриптора безопасности из свойства IADsSecurityDescriptor.DiscretionaryAcl.
Перечислите acEs для объекта и найдите acEs с идентификатором GUID измененного пароля ({AB721A53-1E2F-11D0-9819-00AA0040529B}) для свойства IADsAccessControlEntry.ObjectType и "Все" или "NT AUTHORITY\SELF" для свойства IADsAccessControlEntry.Trustee.
Примечание.
Строки "Все" и "NT AUTHORITY\SELF" локализованы на основе языка первого контроллера домена в домене. Из-за этого строки не следует использовать напрямую. Имена учетных записей должны быть получены во время выполнения путем вызова функции LookupAccountSid с идентификатором безопасности для "Все" ("S-1-1-0") и "NT AUTHORITY\SELF" ("S-1-5-10") известных субъектов безопасности. Примеры функций GetSidAccountName, GetSidAccountName_Everyone и GetSidAccountName_Self C++, показанные в разделе "Чтение пользователя не удается изменить пароль (поставщик LDAP) демонстрирует, как это сделать.
Измените свойство IADsAccessControlEntry.AceType, обнаруженных для ADS_ACETYPE_ACCESS_DENIED_OBJECT, если пользователь не может изменить пароль или ADS_ACETYPE_ACCESS_ALLOWED_OBJECT, если пользователь может изменить пароль.
Если ACE "Все" не найден, создайте новый объект IADsAccessControlEntry, содержащий значения свойств, отображаемые в таблице ниже, и добавьте новую запись в ACL с методом IADsAccessControlList.AddAce.
Если ACE "NT AUTHORITY\SELF" не найден, создайте новый объект IADsAccessControlEntry с теми же значениями свойств, которые отображаются в таблице ниже, кроме свойства Доверенного лица, содержит имя учетной записи для SID "S-1-5-10" ("NT AUTHORITY\SELF"). Добавьте запись в ACL с помощью метода IADsAccessControlList.AddAce.
Чтобы обновить свойство ntSecurityDescriptor объекта, вызовите метод IADs.Put с тем же методом IADsSecurityDescriptor, полученным на шаге 2.
Зафиксируйте локальные изменения на сервере с помощью метода IADs.SetInfo.
Если были созданы какие-либо из acES, необходимо изменить порядок списков ACL таким образом, чтобы acEs находились в правильном порядке. Для этого вызовите функцию GetNamedSecurityInfo с протоколом LDAP ADsPath объекта, а затем функцию SetNamedSecurityInfo с тем же DACL. Эта переупорядочение будет происходить автоматически при добавлении acEs.
В следующей таблице перечислены значения свойств объекта IADsAccessControlEntry.
Свойство IADsAccessControlEntry | Значение |
---|---|
AccessMask | ADS_RIGHT_DS_CONTROL_ACCESS |
AceType | ADS_ACETYPE_ACCESS_DENIED_OBJECT, если пользователь не может изменить пароль или ADS_ACETYPE_ACCESS_ALLOWED_OBJECT, если пользователь может изменить пароль. |
AceFlags | 0 |
Flags | ADS_FLAG_OBJECT_TYPE_PRESENT |
ObjectType | "{AB721A53-1E2F-11D0-9819-00AA0040529B}", который является идентификатором GUID измененного пароля в строковой форме. |
InheritedObjectType | Не используется |
Попечителя | Имя учетной записи для sid "S-1-1-0" (все). |
Пример кода
В следующем примере кода показано, как получить интерфейс для изменения DACL. Интерфейс IADsObjectOptions можно использовать, задав параметр ADS_SECURITY_INFO_DACL.
Примечание.
Чтобы использовать код, описанный в этом примере, необходимо быть Администратор istrator. Если вы не являетесь Администратор istrator, необходимо добавить дополнительный код, который будет использовать интерфейс, который позволит пользователю изменить способ очистки кэша на стороне клиента обратно в службу домен Active Directory.
//
// Obtain an IADsObjectOptions interface from the object whose
// DACL you wish to modify.
//
long CanReadSetDACL = ADS_SECURITY_INFO_DACL;
CComPtr<IADsObjectOptions> spObjOps;
hr = pads->QueryInterface(IID_IADsObjectOptions, (void**)&spObjOps);
if(SUCCEEDED(hr))
{
//
// Set the option mask you want to change. In this case
// we want to change the objects security information, so we'll
// use the ADS_OPTION_SECURITY_MASK. Since we want to modify the
// DACL, we'll set the variant to ADS_SEDCURITY_INFO_DACL flag.
//
CComVariant svar;
svar = CanReadSetDACL;
hr = spObjOps->SetOption(ADS_OPTION_SECURITY_MASK, svar);
}
//
// The smart pointer declared for pObjOptions can be released
// or it will be destroyed and released once the pointer goes
// out of scope.
//
В следующем примере кода показано, как изменить разрешение пользователя на изменение пароля с помощью поставщика LDAP. В этом примере кода используется служебная функция GetObjectACE , определенная выше.
В этом примере используются функции GetSidAccountName_Everyone и GetSidAccountName_Self C++, показанные в разделе "Чтение пользователя не удается изменить пароль (поставщик LDAP)".
#include <aclapi.h>
#define CHANGE_PASSWORD_GUID_W L"{AB721A53-1E2F-11D0-9819-00AA0040529B}"
/***************************************************************************
CreateACE()
Creates an ACE and returns the IDispatch pointer for the ACE. This
pointer can be passed directly to IADsAccessControlList::AddAce.
bstrTrustee - Contains the Trustee for the ACE.
bstrObjectType - Contains the ObjectType for the ACE.
lAccessMask - Contains the AccessMask for the ACE.
lACEType - Contains the ACEType for the ACE.
lACEFlags - Contains the ACEFlags for the ACE.
lFlags - Contains the Flags for the ACE.
***************************************************************************/
IDispatch* CreateACE(BSTR bstrTrustee,
BSTR bstrObjectType,
long lAccessMask,
long lACEType,
long lACEFlags,
long lFlags)
{
HRESULT hr;
IDispatch *pDisp = NULL;
IADsAccessControlEntry *pACE = NULL;
hr = CoCreateInstance(CLSID_AccessControlEntry,
NULL,
CLSCTX_INPROC_SERVER,
IID_IADsAccessControlEntry,
(void**)&pACE);
if(SUCCEEDED(hr))
{
hr = pACE->put_Trustee(bstrTrustee);
hr = pACE->put_ObjectType(bstrObjectType);
hr = pACE->put_AccessMask(lAccessMask);
hr = pACE->put_AceType(lACEType);
hr = pACE->put_AceFlags(lACEFlags);
hr = pACE->put_Flags(lFlags);
hr = pACE->QueryInterface(IID_IDispatch, (LPVOID*)&pDisp);
pACE->Release();
}
return pDisp;
}
/***************************************************************************
ReorderACEs()
Causes the ACEs of an object DACL to be reordered properly. The ACEs are
automatically put in the proper order when they are added to the DACL.
On older systems however, this does not occur automatically, so this
function is necessary so the deny ACEs are ordered in the list before
the allow ACEs.
pwszDN - A null-terminated Unicode string that contains the LDAP
ADsPath of the DS object to reorder the DACL for.
***************************************************************************/
HRESULT ReorderACEs(LPCWSTR pwszDN)
{
HRESULT hr = E_FAIL;
DWORD dwResult;
ACL *pdacl;
PSECURITY_DESCRIPTOR psd;
dwResult = GetNamedSecurityInfoW( (LPWSTR)pwszDN,
SE_DS_OBJECT_ALL,
DACL_SECURITY_INFORMATION,
NULL,
NULL,
&pdacl,
NULL,
&psd);
if(ERROR_SUCCESS == dwResult)
{
dwResult = SetNamedSecurityInfoW( (LPWSTR)pwszDN,
SE_DS_OBJECT_ALL,
DACL_SECURITY_INFORMATION,
NULL,
NULL,
pdacl,
NULL);
LocalFree(psd);
if(ERROR_SUCCESS == dwResult)
{
hr = S_OK;
}
}
return hr;
}
/***************************************************************************
SetUserCannotChangePassword()
Sets the "User Cannot Change Password" permission using the LDAP provider
to the specified setting. To do this, find the existing
ACEs and modify the AceType. If the ACE is not found, a new one of the
proper type is created and added. The ACEs should always be present, but
it is possible that the default DACL excludes them, so this situation
will be handled correctly.
pwszUserDN - A null-terminated Unicode string that contains the LDAP
ADsPath of the user object to modify.
pwszUsername - A null-terminated Unicode string that contains the user
name to use for authorization. If this is NULL, the credentials of the
current user are used.
pwszPassword - A null-terminated Unicode string that contains the
password to use for authorization. This is ignored if pwszUsername is
NULL.
fCannotChangePassword - Contains the new setting for the privilege.
Contains nonzero if the user cannot change their password or zero if
the can change their password.
***************************************************************************/
HRESULT SetUserCannotChangePassword(LPCWSTR pwszUserDN,
LPCWSTR pwszUsername,
LPCWSTR pwszPassword,
BOOL fCannotChangePassword)
{
HRESULT hr;
CComBSTR sbstrEveryone;
hr = GetSidAccountName_Everyone(&sbstrEveryone);
if(FAILED(hr))
{
return hr;
}
CComBSTR sbstrSelf;
hr = GetSidAccountName_Self(&sbstrSelf);
if(FAILED(hr))
{
return hr;
}
if(NULL == pwszUserDN)
{
return E_INVALIDARG;
}
IADs *pads;
hr = ADsOpenObject( pwszUserDN,
pwszUsername,
pwszPassword,
ADS_SECURE_AUTHENTICATION,
IID_IADs,
(LPVOID*)&pads);
if(SUCCEEDED(hr))
{
CComBSTR sbstrSecDesc = "ntSecurityDescriptor";
CComVariant svar;
hr = pads->Get(sbstrSecDesc, &svar);
if(SUCCEEDED(hr))
{
IADsSecurityDescriptor *psd;
hr = svar.pdispVal->QueryInterface(IID_IADsSecurityDescriptor, (LPVOID*)&psd);
if(SUCCEEDED(hr))
{
IDispatch *pDisp;
hr = psd->get_DiscretionaryAcl(&pDisp);
if(SUCCEEDED(hr))
{
IADsAccessControlList *pACL;
hr = pDisp->QueryInterface(IID_IADsAccessControlList, (void**)&pACL);
if(SUCCEEDED(hr))
{
BOOL fMustReorder = FALSE;
/*
Get the existing ACE for the change password permission
for Everyone. If it exists, just modify the existing
ACE. If it does not exist, create a new one and add it
to the ACL.
*/
IADsAccessControlEntry *pACEEveryone = NULL;
hr = GetObjectACE(pACL, CHANGE_PASSWORD_GUID_W, sbstrEveryone, &pACEEveryone);
if(pACEEveryone)
{
hr = pACEEveryone->put_AceType(fCannotChangePassword ?
ADS_ACETYPE_ACCESS_DENIED_OBJECT :
ADS_ACETYPE_ACCESS_ALLOWED_OBJECT);
pACEEveryone->Release();
}
else
{
IDispatch *pDispEveryone = NULL;
pDispEveryone = CreateACE(sbstrEveryone,
CComBSTR(CHANGE_PASSWORD_GUID_W),
ADS_RIGHT_DS_CONTROL_ACCESS,
fCannotChangePassword ?
ADS_ACETYPE_ACCESS_DENIED_OBJECT :
ADS_ACETYPE_ACCESS_ALLOWED_OBJECT,
0,
ADS_FLAG_OBJECT_TYPE_PRESENT);
if(pDispEveryone)
{
//add the new ACE for everyone
hr = pACL->AddAce(pDispEveryone);
pDispEveryone->Release();
fMustReorder = TRUE;
}
}
/*
Get the existing ACE for the change password permission
for Self. If it exists, just modify the existing
ACE. If it does not exist, create a new one and add it
to the ACL.
*/
IADsAccessControlEntry *pACESelf = NULL;
hr = GetObjectACE(pACL, CHANGE_PASSWORD_GUID_W, sbstrSelf, &pACESelf);
if(pACESelf)
{
hr = pACESelf->put_AceType(fCannotChangePassword ?
ADS_ACETYPE_ACCESS_DENIED_OBJECT :
ADS_ACETYPE_ACCESS_ALLOWED_OBJECT);
pACESelf->Release();
}
else
{
IDispatch *pDispSelf = NULL;
pDispSelf = CreateACE(sbstrSelf,
CComBSTR(CHANGE_PASSWORD_GUID_W),
ADS_RIGHT_DS_CONTROL_ACCESS,
fCannotChangePassword ?
ADS_ACETYPE_ACCESS_DENIED_OBJECT :
ADS_ACETYPE_ACCESS_ALLOWED_OBJECT,
0,
ADS_FLAG_OBJECT_TYPE_PRESENT);
if(pDispSelf)
{
//add the new ACE for self
hr = pACL->AddAce(pDispSelf);
pDispSelf->Release();
fMustReorder = TRUE;
}
}
//update the security descriptor property
hr = pads->Put(sbstrSecDesc, svar);
//commit the changes
hr = pads->SetInfo();
if(fMustReorder)
{
ReorderACEs(pwszUserDN);
}
pACL->Release();
}
pDisp->Release();
}
psd->Release();
}
}
}
return hr;
}
В следующем примере кода показано, как изменить разрешение пользователя на изменение пароля с помощью поставщика LDAP.
Примечание.
Приведенный ниже пример будет работать только на доменах, где основной язык является английским, так как строки "Все" и "NT AUTHORITY\SELF" локализованы на основе языка первого контроллера домена в домене. В Visual Basic нет способа получить имена учетных записей для известного субъекта безопасности без вызова функции LookupAccountSid. При использовании Visual Basic рекомендуется использовать поставщик WinNT для изменения разрешения пользователя на изменение пароля, как показано в разделе "Изменение пароля пользователя не удается изменить пароль( поставщик WinNT).
'******************************************************************************
'
' SetUserCannotChangePassword
'
' Sets the "User Cannot Change Password" permission using the LDAP provider
' to the specified setting. This is accomplished by finding the existing
' ACEs and modifying the AceType. The ACEs should always be present, but
' it is possible that the default DACL excludes them. This function will not
' work correctly if both ACEs are not present.
'
' strUserDN - A string that contains the LDAP ADsPath of the user object to
' modify.
'
' strUsername - A string that contains the user name to use for
' authorization. If this is an empty string, the credentials of the current
' user are used.
'
' strPassword - A string that contains the password to use for authorization.
' This is ignored if strUsername is empty.
'
' fCannotChangePassword - Contains the new setting for the privilege.
' Contains True if the user cannot change their password or False if
' the can change their password.
'
'******************************************************************************
Sub SetUserCannotChangePassword(strUserDN As String, strUsername As String, strPassword As String, fUserCannotChangePassword As Boolean)
Dim oUser As IADs
Dim oSecDesc As IADsSecurityDescriptor
Dim oACL As IADsAccessControlList
Dim oACE As IADsAccessControlEntry
fEveryone = False
fSelf = False
If "" <> strUsername Then
Dim dso As IADsOpenDSObject
' Bind to the group with the specified user name and password.
Set dso = GetObject("LDAP:")
Set oUser = dso.OpenDSObject(strUserDN, strUsername, strPassword, 1)
Else
' Bind to the group with the current credentials.
Set oUser = GetObject(strUserDN)
End If
Set oSecDesc = oUser.Get("ntSecurityDescriptor")
Set oACL = oSecDesc.DiscretionaryAcl
' Modify the existing entries.
For Each oACE In oACL
If UCase(oACE.ObjectType) = UCase(CHANGE_PASSWORD_GUID) Then
If oACE.Trustee = "Everyone" Then
' Modify the ace type of the entry.
If fUserCannotChangePassword Then
oACE.AceType = ADS_ACETYPE_ACCESS_DENIED_OBJECT
Else
oACE.AceType = ADS_ACETYPE_ACCESS_ALLOWED_OBJECT
End If
End If
If oACE.Trustee = "NT AUTHORITY\SELF" Then
' Modify the ace type of the entry.
If fUserCannotChangePassword Then
oACE.AceType = ADS_ACETYPE_ACCESS_DENIED_OBJECT
Else
oACE.AceType = ADS_ACETYPE_ACCESS_ALLOWED_OBJECT
End If
End If
End If
Next
' Update the ntSecurityDescriptor property.
oUser.Put "ntSecurityDescriptor", oSecDesc
' Commit the changes to the server.
oUser.SetInfo
End Sub