c# - 使用 AD 的服务帐户和用户帐户凭据连接到 Active Directory 以登录(在我的产品中)
问题描述
我目前正在使用 Directory Searcher 对 AD 用户进行身份验证。
DirectoryEntry adsEntry = new DirectoryEntry(ConfigurationManager.AppSettings["ADConnectionString"], username, password, System.DirectoryServices.AuthenticationTypes.Secure);
DirectorySearcher adsSearcher = new DirectorySearcher(adsEntry);
adsSearcher.Filter = "(sAMAccountName=" + _userName + ")";
SetPropertiesToLoad(ref adsSearcher);
SearchResult adsSearchResult = adsSearcher.FindOne();
Logger.Debug("After adsSearcher.FindOne() success");
if (!ExtractPropertiesReceivedFromAD(adsSearchResult, ref emailAddress, ref _name, username, ref errorMessage))
return false;
这适用于许多 AD 设置,但最近我遇到了 AD 中的 1 个不允许连接到它。
我的客户说他们有 LDAP 身份验证,所以我不能在不提供服务帐户凭据的情况下直接查询 AD。
因此,在这种情况下,要使用 LDAP 与 AD 连接,我需要 1 个凭据,然后发布以验证用户身份,我需要他自己的用户名/密码。
现在我如何在 DirectorySearcher 中适应这种情况?
解决方案
这是我用来使用系统凭据绑定到 LDAP 目录、搜索提供的用户 ID,然后验证提供的用户凭据的函数。要将函数与 Active Directory 一起使用,“strUIDAttr”是 sAMAccountName。
protected string ldapAuthTest(string strLDAPServer, string strSuppliedUser, string strSuppliedPwd, string strSystemUID, string strSystemPwd, string strLDAPUserBase, string strUIDAttr)
{
strSuppliedUser = strSuppliedUser.Trim();
string strResults = "";
string strLDAPUserHost = strLDAPServer + strLDAPUserBase;
// Establish LDAP connection and bind with system ID
System.DirectoryServices.DirectoryEntry dirEntry = new System.DirectoryServices.DirectoryEntry();
dirEntry.Path = strLDAPUserHost;
dirEntry.Username = strSystemUID;
dirEntry.Password = strSystemPwd;
//dirEntry.AuthenticationType = System.DirectoryServices.AuthenticationTypes.None;
dirEntry.AuthenticationType = System.DirectoryServices.AuthenticationTypes.SecureSocketsLayer;
try
{
dirEntry.RefreshCache();
// Search directory for the user logging on
string strLDAPFilter = "(&(" + strUIDAttr + "=" + strSuppliedUser + "))";
System.DirectoryServices.DirectorySearcher ldapSearch = new System.DirectoryServices.DirectorySearcher(dirEntry);
ldapSearch.ServerTimeLimit = new TimeSpan(0, 0, 30);
ldapSearch.Filter = strLDAPFilter;
ldapSearch.SearchScope = System.DirectoryServices.SearchScope.Subtree;
System.DirectoryServices.SearchResultCollection searchResults = ldapSearch.FindAll();
if (searchResults.Count == 1)
{
string strLogonUserBase = searchResults[0].GetDirectoryEntry().Path;
// get rid of strLDAPServer from directory entry path
string strLogonUserFQDN = strLogonUserBase.Replace(strLDAPServer, "");
dirEntry.Close();
// Attempt to bind as the user
System.DirectoryServices.DirectoryEntry userAuthAttempt = new System.DirectoryServices.DirectoryEntry();
userAuthAttempt.Path = strLDAPUserHost;
userAuthAttempt.Username = strLogonUserFQDN;
userAuthAttempt.Password = strSuppliedPwd;
//userAuthAttempt.AuthenticationType = System.DirectoryServices.AuthenticationTypes.None;
userAuthAttempt.AuthenticationType = System.DirectoryServices.AuthenticationTypes.SecureSocketsLayer;
try
{
userAuthAttempt.RefreshCache();
userAuthAttempt.Close();
strResults = "<td><font color='green'>User " + UserName.Value + " has authenticated successfully.</font></td>";
}
catch (Exception e)
{
string strAuthError = e.Message;
string strLockedOut = "A constraint violation occurred.\r\n";
string strBadPwd = "Logon failure: unknown user name or bad password.\r\n";
string strNSAccountLock = "The server is unwilling to process the request.\r\n";
if (String.Compare(strAuthError, strBadPwd) == 0)
strResults = "<td><font color='red'>Logon failure for user " + UserName.Value + " - password is invalid.</font></td></tr>"; ;
}
else if (String.Compare(strAuthError, strLockedOut) == 0)
{
strResults = "<td><font color='red'>Logon failure for user " + UserName.Value + " - account is locked out.</font></td>"; ;
}
else if (String.Compare(strAuthError, strNSAccountLock) == 0)
{
strResults = "<td><font color='red'>Logon failure for user " + UserName.Value + " - password has expired.</font></td>"; ;
}
else
{
strResults = "<td><font color='red'>Logon failure for user " + UserName.Value + " (" + strLogonUserFQDN + ") :" + strAuthError + "</font></td>"; ;
}
}
}
else if (searchResults.Count > 1)
{
strResults = "<td><font color='red'>Account " + UserName.Value + " was found in the directory " + searchResults.Count + " times. Please contact the Help Desk to have this issue corrected.</font></td>"; ;
}
else
{
strResults = "<td><font color='red'>Account " + UserName.Value + " was not found in the directory.</font></td>"; ;
}
return strResults;
}
catch(Exception e)
{
string strAuthError = e.Message;
string strConnectFail = "The server is not operational.\r\n";
if (String.Compare(strAuthError, strConnectFail) == 0)
{
strResults = "<td><font color='red'>Transient connection failure, please try again.</font></td>"; ;
}
else
{
strResults = "<td><font color='red'>Transient failure (" + strAuthError + "), please try again.</font></td>";
}
return strResults;
}
}
推荐阅读
- android - React Native:安装到虚拟/物理设备失败
- php - 友好的 url 不识别变量
- c++ - 使用 Gtk3 检测屏幕上任意位置的鼠标点击
- flutter - Flutter:仅适用于 Web 用户的导入包
- google-cloud-platform - 将项目 (VM) 迁移到另一个 GCP 帐户以使用免费层
- c++ - 为什么我的结构占用的内存比请求的多?
- javascript - How do I make an array with inputs and then pass it to another function jQuery
- php - 数组中的值?
- javascript - Getting null value when interpolating in querySelector - works when hard-coded
- node.js - npm 安装错误:找不到模块'read-package-tree'