我设置LDAP DirContext.search(...)来忽略引用, 但是当我调用NamingEnumeration.hasMore()时,我仍然会得到引用异常。
Exception in thread "main" javax.naming.PartialResultException: Unprocessed Continuation Reference(s); remaining name 'DC=company,DC=com'
at com.sun.jndi.ldap.LdapCtx.processReturnCode(LdapCtx.java:2846)
at com.sun.jndi.ldap.LdapCtx.processReturnCode(LdapCtx.java:2820)
at com.sun.jndi.ldap.LdapNamingEnumeration.getNextBatch(LdapNamingEnumeration.java:129)
at com.sun.jndi.ldap.LdapNamingEnumeration.hasMoreImpl(LdapNamingEnumeration.java:198)
at com.sun.jndi.ldap.LdapNamingEnumeration.hasMore(LdapNamingEnumeration.java:171)
我可以告诉DirContext.search忽略引荐, 这样NamingEnumeration.hasMore()返回false而不是抛出异常?
这是剪辑:
import javax.naming.*;
import javax.naming.directory.*;
Properties p = new Properties();
p.setProperty(Context.INITIAL_CONTEXT_FACTORY, ldapInitContextFactory);
p.setProperty(Context.PROVIDER_URL, ldapURL);
p.setProperty(Context.SECURITY_CREDENTIALS, ldapPassword);
p.setProperty(Context.SECURITY_PRINCIPAL, ldapUser);
p.setProperty(Context.SECURITY_AUTHENTICATION, "simple");
p.setProperty(Context.REFERRAL, "ignore");
DirContext ctx = new InitialDirContext(p);
SearchControls searchControls = new SearchControls();
searchControls.setSearchScope(SearchControls.SUBTREE_SCOPE);
searchControls.setDerefLinkFlag(false);
NamingEnumeration e = ctx.search(ldapBaseDN, ldapQuery, null, searchControls);
for (; e.hasMore();) {
SearchResult sr = (SearchResult) e.next();
System.out.println("\nSearch Result: " + sr.getName());
}
注意:如果我启用了引用,当我调用NamingEnumeration.hasMore()时,我会得到一个LdapReferralException expcetion。
答案 0 :(得分:7)
javax/naming/NamingEnumeration.java
public interface NamingEnumeration<T> extends Enumeration<T> {
public boolean hasMore() throws NamingException;
public T next() throws NamingException;
...
}
java/util/Enumeration.java
public interface Enumeration<E> {
boolean hasMoreElements();
E nextElement();
}
调用e.hasMoreElements()而不是e.hasMore()解决了这个问题。 也就是说,当有引用时,它返回false(而不是抛出异常)。
但是,它不是最佳解决方案,因为我们可能会遗漏其他真正的NamingException异常(例如CommunicationException)。
我仍然想找到告诉DirContext.search忽略推荐的正确方法, 这样NamingEnumeration.hasMore()返回false而不是抛出异常。 任何想法?
我在http://download.java.net/openjdk/jdk6/下载了JDK源代码 - &gt;的openjdk-6-SRC-b24-14_nov_2011.tar.gz 但是这个源代码与JDK二进制文件不完全对应, 见how to find the exact sources of a JDK1.6 binary (including com.sun.jndi.*)
从下面的JDK源代码中,当有引用时,它似乎不可能得到“假”而不是异常。
./jdk/src/share/classes/com/sun/jndi/ldap/LdapCtx.java
protected void processReturnCode(LdapResult res, Name resolvedName, Object resolvedObj, Name remainName, Hashtable envprops, String fullDN) throws NamingException {
NamingException e;
switch (res.status) {
case LdapClient.LDAP_SUCCESS:
// handle Search continuation references
if (res.referrals != null) {
msg = "Unprocessed Continuation Reference(s)";
if (handleReferrals == LdapClient.LDAP_REF_IGNORE) {
e = new PartialResultException(msg);
break;
}
[...]
}
[...]
throw e;
}
但我仍然不确定。
我认为我们可以告诉DirContext.search忽略推荐是有意义的, 这样NamingEnumeration.hasMore()返回false而不是抛出异常。
任何想法?
答案 1 :(得分:1)
据我所知, 是一种忽略引用的方法,这种方法在处理Active Directory LDAP操作时特别有用。您必须创建并注入新的LdapTemplate
(由配置好的ContextSource
构成),而不是从头开始创建自己的InitialDirContext
。我无法确定这是否与您的设置兼容。我希望这会有所帮助。
从Spring LDAP核心LdapTemplate
documentation:
AD服务器显然无法自动处理引荐,这会导致在搜索中遇到引荐时抛出
PartialResultException
。为避免这种情况,请将ignorePartialResultException
属性设置为true
。