Java-实现LDAP认证(获取用户信息)

(可选)下载一个ldap浏览器,http://www.ldapbrowserwindows.com/

起个名字,下一步

填写好相关信息
账号密码
Filter是过滤信息,这里默认就好了,因为怕进去误删数据,我这里选择只查看
可以看到进去后它是一个目录结构,打开可以看到它的信息

前面我们已经测试了地址可以连进去并且看到了信息,下面是工具类代码(因为公司业务需求,所以写了这个工类,小伙伴们可以忽略),lombok这个包可以不导:

import java.util.Hashtable;
import javax.naming.Context;
import javax.naming.NamingException;
import javax.naming.ldap.InitialLdapContext;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Component;
import lombok.extern.slf4j.Slf4j;

@Component
@Slf4j
public class LDAPUtil {

    @Value("${ldapUrl}")
    private String LDAPURL;
    @Value("${BaseDN}")
    private String BASEDN;

    public boolean connectLDAP(String userName, String passwd, String SearchName) {
        Hashtable<String, String> env = new Hashtable<String, String>();
        log.debug("===" + userName + "开始认证LDAP===");
        log.debug("password:" + passwd);

        boolean result = false;

        env.put(Context.SECURITY_PRINCIPAL, "uid=" + userName + "," + SearchName);//用户名
        log.debug("uid=" + userName + "," + SearchName);
        env.put(Context.SECURITY_CREDENTIALS, passwd);//密码
        env.put(Context.PROVIDER_URL, LDAPURL + BASEDN);//连接LDAP的URL和端口(这里的BASEDN你们可以不用,只要LDAPURL)
        env.put(Context.INITIAL_CONTEXT_FACTORY, "com.sun.jndi.ldap.LdapCtxFactory");//JNDI Context工厂类
        env.put(Context.SECURITY_AUTHENTICATION, "simple");//认证类型

        try {
            new InitialLdapContext(env, null);//开始连接
            result = true;
            log.debug("===认证成功===");
        } catch (NamingException e) {
            log.debug("===认证失败===");
        }
        return result;
    }
}

工具类写好之后我们写实现:

import java.util.HashMap;
import java.util.Hashtable;
import java.util.Map;
import javax.naming.Context;
import javax.naming.NamingEnumeration;
import javax.naming.NamingException;
import javax.naming.directory.Attribute;
import javax.naming.directory.SearchControls;
import javax.naming.directory.SearchResult;
import javax.naming.ldap.InitialLdapContext;
import javax.naming.ldap.LdapContext;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Component;
import lombok.extern.slf4j.Slf4j;

/**
 * 
 * @version:
 * @Description: LDAP认证获取用户信息
 * @author: huyw
 * @date: 2018年10月5日 下午1:15:09
 */
@Slf4j
@Component
public class LADPGetUser {

    @Value("${BaseDN}")
    private String BASEDN;

    @Value("${pwd}")
    private String PASSWORD;

    @Value("${ldapUrl}")
    private String URL;

    @Value("${staffSearchName}")
    private String STAFFSEARCHNAME;

    @Value("${gstudentSearchName}")
    private String GSTUDENTSEARCHNAME;

    @Value("${studentSearchName}")
    private String STUDENTSEARCHNAME;

    @Value("${principle}")
    private String PRINCIPLE;

    @Autowired
    private LDAPUtil ldapUtil;

    public Map<String, Object> getUser(String uid, String pwd) {
        
        Map<String, Object> map = new HashMap<String, Object>();
        try {
            //连接LDAP
            LdapContext ctx = connetLDAP();
            //过滤条件
            String filter = "(&(objectClass=*)(uid=" + uid + "))";
            //要获取的字段信息
            String[] attrPersonArray = { "uid", "userPassword", "displayName", "cn", "sn", "mail", "description" };
            SearchControls searchControls = new SearchControls();//搜索控件
            searchControls.setSearchScope(2);//搜索范围
            searchControls.setReturningAttributes(attrPersonArray);
            //1.要搜索的上下文或对象的名称;2.过滤条件,可为null,默认搜索所有信息;3.搜索控件,可为null,使用默认的搜索控件
            NamingEnumeration<SearchResult> answer = ctx.search("ou=People,dc=uestc,dc=edu,dc=cn", filter.toString(),searchControls);

            while (answer.hasMore()) {
                SearchResult result = (SearchResult) answer.next();
                NamingEnumeration attrs = result.getAttributes().getAll();
                while (attrs.hasMore()) {
                    Attribute attr = (Attribute) attrs.next();
                    log.debug(attr.getID() + "=" + attr.get());
                    map.put(attr.getID(), attr.get());
                }
            }
            // 在校研究生
            boolean flag = ldapUtil.connectLDAP(uid, pwd, GSTUDENTSEARCHNAME);
            if (!flag) {
                // 在校本科生
                flag = ldapUtil.connectLDAP(uid, pwd, STUDENTSEARCHNAME);
                if (!flag) {
                    // 在校教职工
                    flag = ldapUtil.connectLDAP(uid, pwd, STAFFSEARCHNAME);
                }
            }
            map.put("flag", Boolean.valueOf(flag));

        } catch (Exception e) {
            log.error("===认证失败===");
        }

        return map;
    }

    public LdapContext connetLDAP() throws NamingException {

        log.debug("====管理员开始连接====");

        Hashtable<String, Object> env = new Hashtable<String, Object>();
        env.put(Context.SECURITY_PRINCIPAL, PRINCIPLE);//用户名
        env.put(Context.SECURITY_CREDENTIALS, PASSWORD);//密码
        env.put(Context.PROVIDER_URL, URL);//LDAP的地址:端口
        env.put(Context.INITIAL_CONTEXT_FACTORY, "com.sun.jndi.ldap.LdapCtxFactory");//LDAP工厂类
        env.put(Context.SECURITY_AUTHENTICATION, "simple");//认证类型
        LdapContext ctxTDS = new InitialLdapContext(env, null);//连接

        return ctxTDS;
    }
}

这个是我的配置文件:

#project
project.name=
project.packageName=

#ldap
Host=xxxxxx
Port=389
BaseDN=dc=xx,dc=xx,dc=xxx
principle=uid=xxxx,ou=xxxx,dc=xxx,dc=xxx,dc=xxx
pwd=xxx
ldapUrl=LDAP://xxxxxxxxx:389/

#搜索属性
#在校教职工所在ou:
#其实就是所在目录
staffSearchName=ou=xxx,ou=People,dc=uestc,dc=edu,dc=cn
#在校本科生所在ou:
studentSearchName=ou=xxx,ou=People,dc=uestc,dc=edu,dc=cn
#在校研究生所在ou:
gstudentSearchName=ou=xxx,ou=People,dc=uestc,dc=edu,dc=cn

LADP工具类:


import java.util.Hashtable;

import javax.naming.Context;
import javax.naming.NamingEnumeration;
import javax.naming.NamingException;
import javax.naming.directory.Attribute;
import javax.naming.directory.SearchControls;
import javax.naming.directory.SearchResult;
import javax.naming.ldap.InitialLdapContext;
import javax.naming.ldap.LdapContext;

import com.alibaba.fastjson.JSONObject;

/**
 * ldap连接
 * 
 * @version:
 * @Description:
 * @author: huyw
 * @date: 2018年12月24日 上午11:27:38
 */
public class LdapUtils {

    /**
     * 管理员连接
     * 
     * @Description:
     * @param principle
     * @param password
     * @param url
     * @return
     */
    public static LdapContext connectLdapAdmin(String principle, String password, String url) {
        LdapContext ctxTDS = null;
        Hashtable<String, Object> env = new Hashtable<String, Object>();
        env.put(Context.SECURITY_PRINCIPAL, principle);
        env.put(Context.INITIAL_CONTEXT_FACTORY, "com.sun.jndi.ldap.LdapCtxFactory");
        env.put(Context.SECURITY_CREDENTIALS, password);
        env.put(Context.PROVIDER_URL, url);
        env.put(Context.INITIAL_CONTEXT_FACTORY, "com.sun.jndi.ldap.LdapCtxFactory");
        env.put(Context.SECURITY_AUTHENTICATION, "simple");
        try {
            ctxTDS = new InitialLdapContext(env, null);
        } catch (NamingException e) {
            e.printStackTrace();
        }
        return ctxTDS;
    }

    /**
     * ldap
     */
    public static boolean connectLdap(String userName, String password, String SearchName, String ldapUrl,
            String baseDN) {
        Hashtable<String, String> env = new Hashtable<String, String>();
        boolean result = false;
        env.put(Context.SECURITY_PRINCIPAL, "uid=" + userName + "," + SearchName);
        env.put(Context.SECURITY_CREDENTIALS, password);
        env.put(Context.PROVIDER_URL, ldapUrl + baseDN);
        env.put(Context.INITIAL_CONTEXT_FACTORY, "com.sun.jndi.ldap.LdapCtxFactory");
        env.put(Context.SECURITY_AUTHENTICATION, "simple");
        try {
            new InitialLdapContext(env, null);
            result = true;
        } catch (NamingException e) {
            System.out.println("userName:" + userName + ",SearchName:" + SearchName + "登录失败");
        }
        return result;
    }

    public static void main(String[] args) {
        String certId = "";
        String password = "";
        String principle = "";
        String passwordLdap = "";
        String url = "";
        String teacherSearch = "";
        String studentSearch = "";
        String baseDN = "";
        try {
            LdapContext ctx = LdapUtils.connectLdapAdmin(principle, passwordLdap, url);
            String filter = "(&(objectClass=*)(uid=" + certId + "))";
            String[] attrPersonArray = { "uid", "employeeType", "displayName"};
            SearchControls searchControls = new SearchControls();
            searchControls.setSearchScope(2);
            searchControls.setReturningAttributes(attrPersonArray);
            NamingEnumeration<SearchResult> answer = ctx.search("ou=Users,dc=test,dc=edu,dc=cn", filter.toString(),
                    searchControls);
            JSONObject resultJson = new JSONObject();
            while (answer.hasMore()) {
                SearchResult result = (SearchResult) answer.next();
                NamingEnumeration attrs = result.getAttributes().getAll();
                while (attrs.hasMore()) {
                    Attribute attr = (Attribute) attrs.next();
                    resultJson.put(attr.getID(), attr.get());
                }
            }
            // 在校学生
            boolean studentFlag = LdapUtils.connectLdap(certId, password, studentSearch, url, baseDN);
            if (studentFlag) {
                System.out.println("ldap用户信息:" + resultJson.toJSONString());
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 199,636评论 5 468
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 83,890评论 2 376
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 146,680评论 0 330
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 53,766评论 1 271
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 62,665评论 5 359
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,045评论 1 276
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,515评论 3 390
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,182评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,334评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,274评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,319评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,002评论 3 315
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,599评论 3 303
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,675评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,917评论 1 255
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,309评论 2 345
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 41,885评论 2 341