Пароль сброса LDAP из-за пределов доменной сети C# Ошибка: RP C сервер недоступен. (исключение из hresult: 0x800706ba) - PullRequest
3 голосов
/ 29 мая 2020

Мы пытаемся сбросить пароль LDAP , он работает в среде разработки, но не работает в производственной среде.

Наша среда разработки находится внутри домена, а производственная среда - за пределами домена .

При разработке для подключения LDAP мы использовали доменное имя, например abc.com, и производственную среду, которую мы используем IPaddress:389, которая уже работает для LDAP аутентификации пользователя в обеих средах. Но не работает для Пароль сброса LDAP .

Error: RPC server is unavailable. (exception from hresult: 0x800706ba)

Разработка: (работает)

PrincipalContext principalContext =
    new PrincipalContext(ContextType.Domain, "<domain.com>", container: "<DC=domain,DC=com>",
    "<username>", "<password>");
UserPrincipal user = UserPrincipal.FindByIdentity(principalContext, "<LdapUserName>");
// "<username>", "<password>" are Administrative credential.
bool isValid = user.ValidateCredentials("<username>", "<password>");
_logger.Log($"Is Connection: {isValid}");
**// Output: Is Connection: True**
user.UserCannotChangePassword = false;
user.SetPassword("<NewPassword>");
// Password has been successfully reset.

Производство: (работает ) Также мы аутентифицируем пользователей LDAP, используя следующий метод, который он работает на производстве:

Проверяем, есть ли у пользователя учетная запись LDAP:

// "<username>", "<password>" are Administrative credential.
var entry = new DirectoryEntry($"LDAP://{"<IP:389>"}", "<username>", "<password>",
    AuthenticationTypes.Secure | AuthenticationTypes.Sealing | AuthenticationTypes.ServerBind);
var search = new DirectorySearcher(entry);
var strFilter = $"(mail={"<UserEmailId>"})";
search.Filter = strFilter;
var result = await Task.Run(() => search.FindOne());
if (result != null)
{
    //IsLdapUser = true;
    //result.Properties["samaccountname"][0]);
}
else
{
    //IsLdapUser = false;
}
// Successfully


// Authenticate LDAP user:
var ldapConnection = new LdapConnection(new LdapDirectoryIdentifier("<IP:389>", false, false));                    
var nc = new NetworkCredential("<LdapUserName>", "<LdapUserPassword>", "<IP:389>");
ldapConnection.Credential = nc;
ldapConnection.AuthType = AuthType.Negotiate;
ldapConnection.Bind(nc);
// Successfully

Производство: (не работает)

// "<username>", "<password>" are Administrative credential.
PrincipalContext principalContext =
    new PrincipalContext(ContextType.Domain, "<IP:389>", container: "<DC=domain,DC=com>",
    "<username>", "<password>");
UserPrincipal user = UserPrincipal.FindByIdentity(principalContext, "<LdapUserName>");
bool isValid = user.ValidateCredentials("<username>", "<password>");
_logger.Log($"Is Connection: {isValid}");
**// Output: Is Connection: True**
user.UserCannotChangePassword = false;
user.SetPassword("<NewPassword>");
// Error: RPC server is unavailable. (exception from hresult: 0x800706ba)

Также пробовал с кодом ниже (не работает)

// "<username>", "<password>" are Administrative credential.
DirectoryEntry de = new DirectoryEntry("<IP:389>","<username>", "<password>", 
AuthenticationTypes.Secure | AuthenticationTypes.Sealing | AuthenticationTypes.ServerBind);
// LDAP Search Filter
DirectorySearcher ds = new DirectorySearcher(de);
ds.Filter = "(&(objectClass=user)(|(sAMAccountName=" + "<LdapUserName>"+ ")))";

// LDAP Properties to Load
ds.PropertiesToLoad.Add("displayName");
ds.PropertiesToLoad.Add("sAMAccountName");
ds.PropertiesToLoad.Add("DistinguishedName");
ds.PropertiesToLoad.Add("CN");

// Execute Search
SearchResult result = await Task.Run(() => ds.FindOne());

string dn = result.Properties["DistinguishedName"][0].ToString();

DirectoryEntry uEntry = result.GetDirectoryEntry();

uEntry.Invoke("SetPassword", new object[] { "<NewPassword>"});  //Set New Password                        
uEntry.CommitChanges();
uEntry.Close();
// Error: RPC server is unavailable. (exception from hresult: 0x800706ba)

1 Ответ

1 голос
/ 29 мая 2020

Атрибут, используемый для изменения пароля: unicodePwd. Эта документация раскрывает некоторые условия, которые должны быть выполнены для изменения пароля. Прежде всего, соединение должно быть зашифровано.

Вызов .Invoke("SetPassword", ...) фактически вызывает собственный метод Windows IADsUser::SetPassword. Эта документация показывает, что он автоматически пытается использовать несколько различных способов шифрования. Исключение происходит из-за того, что ни один из этих методов не сработал.

Фактически вы можете изменить атрибут unicodePwd напрямую, не вызывая SetPassword, к которому я доберусь, но, тем не менее, вы должны решить проблему сначала шифрование.

Когда вы запускаете это с компьютера внутри сети, AuthenticationTypes.Sealing будет достаточно. Как в документации указано , эффект заключается в том, что он использует Kerberos для шифрования соединения.

Но когда вы подключаетесь извне домена, Kerberos не будет работать (возможно, он будет работать с усилия - я не эксперт по Kerberos). Таким образом, единственный используемый метод шифрования - это SSL. Метод SetPassword на самом деле пытается использовать SSL, но явно не работает.

Одна проблема, которую я сразу вижу, заключается в том, что вы используете IP-адрес для подключения к D C, и SSL не будет работать с IP-адресом, поскольку имя домена в сертификате SSL должно совпадать с именем, которое вы используете для доступа к серверу, а на сертификате SSL не будет IP-адреса. Поэтому вам придется изменить это, чтобы использовать доменное имя. Если DNS не разрешит имя, вы можете добавить его в свой файл hosts.

Изменение может исправить все. В противном случае могут быть две другие проблемы:

  1. Порт 636 недоступен (мешает брандмауэр).
  2. Сертификат SSL не является доверенным на компьютере, на котором вы его запускаете. на

LDAP через SSL (LDAPS) работает на порту 636. Вы можете проверить это соединение в PowerShell:

Test-NetConnection example.com -Port 636

Если это не удается, сначала исправьте это.

Далее проверьте сертификат. Вы можете загрузить сертификат с помощью этого сценария PowerShell:

$webRequest = [Net.WebRequest]::Create("https://example.com:636")
try { $webRequest.GetResponse() } catch {}
$cert = $webRequest.ServicePoint.Certificate
$bytes = $cert.Export([Security.Cryptography.X509Certificates.X509ContentType]::Cert)
set-content -value $bytes -encoding byte -path "certificate.cer"

Измените example.com в первой строке на свое доменное имя (оставьте https:// и :636). Тогда у вас будет файл с именем certificate.cer в текущем каталоге, который вы можете открыть и проверить. Он предупредит вас, если ему не доверяют. Если он не является доверенным, вам нужно будет установить сертификат root на сервере в качестве доверенного сертификата Root.

Если он уже является доверенным, убедитесь, что доменное имя «Кому выдано:» на сертификат соответствует имени, которое вы использовали для подключения. В нашей среде сертификаты SSL указываются в имени каждого контроллера домена (dc1.example.com), а не в имени домена (example.com). Поэтому мне нужно настроить таргетинг на конкретный контроллер домена c, чтобы LDAPS работал.

Как только вы разберетесь со всем этим, ваш код должен работать.

Если вы хотите изменить * Атрибут 1056 * напрямую вместо использования SetPassword (который может работать или не работать немного быстрее), вам нужно будет установить исходное соединение через SSL. Например:

DirectoryEntry de = new DirectoryEntry("LDAP://dc1.example.com:636","<username>", "<password>", 
    AuthenticationTypes.Secure | AuthenticationTypes.SecureSocketsLayer | AuthenticationTypes.ServerBind);

Используйте AuthenticationTypes.ServerBind только в том случае, если вы нацеливаетесь на конкретный c D C.

Затем вы можете обновить атрибут unicodePwd в самом конкретном c так, как он хочет:

uEntry.Properties["unicodePwd"].Value = Encoding.Unicode.GetBytes("\"NewPassword\"");
uEntry.CommitChanges();

Обратите внимание, что новый пароль должен быть заключен в кавычки.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...