1. 程式人生 > >java LDAP 使用 DirSyncControl 對windows AD進行變更輪詢

java LDAP 使用 DirSyncControl 對windows AD進行變更輪詢

import com.sun.jndi.ldap.ctl.DirSyncControl;
import com.sun.jndi.ldap.ctl.DirSyncResponseControl;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import javax.naming.Context;
import javax.naming.NamingEnumeration;
import javax.naming.NamingException;
import javax.naming.directory.Attribute;
import javax.naming.directory.Attributes;
import javax.naming.directory.SearchControls;
import javax.naming.directory.SearchResult;
import javax.naming.ldap.Control;
import javax.naming.ldap.InitialLdapContext;
import javax.naming.ldap.LdapContext;
import java.io.IOException;
import java.util.Hashtable;
import java.util.concurrent.TimeUnit;

public class Demo {

    private static Logger logger = LoggerFactory.getLogger(Demo.class);

    public static void main(String[] args) throws InterruptedException, NamingException, IOException {
        modificationDetect();
    }

    /**
     * @throws NamingException
     * @throws IOException
     * @throws InterruptedException
     */
    public static void modificationDetect() throws NamingException, IOException, InterruptedException {

        //連線配置
        Hashtable<Object, Object> env = new Hashtable<>();
        env.put(Context.INITIAL_CONTEXT_FACTORY, "com.sun.jndi.ldap.LdapCtxFactory");
        env.put(Context.SECURITY_AUTHENTICATION, "simple");
        env.put(Context.SECURITY_PRINCIPAL, "
[email protected]
"); env.put(Context.SECURITY_CREDENTIALS, "123"); env.put(Context.PROVIDER_URL, "ldap://172.16.1.222:389"); LdapContext ctx = new InitialLdapContext(env, null); SearchControls searchCtls = new SearchControls(); searchCtls.setSearchScope(SearchControls.SUBTREE_SCOPE); //搜尋過濾 String searchFilter = "(&(objectClass=user)(objectCategory=person))"; String searchBase = "DC=owo,DC=com"; byte[] cookie; int totalResults = 0; int page = 0; //第一次不傳cookie ctx.setRequestControls(new Control[]{new DirSyncControl()}); NamingEnumeration answer = ctx.search(searchBase, searchFilter, searchCtls); cookie = retrieveCookie(ctx); //windows ad 預設單次結果返回1000條記錄,所以得不斷遍歷 while (answer.hasMoreElements()) { page++; while (answer.hasMoreElements()) { SearchResult entry = (SearchResult) answer.next(); totalResults++; logger.info("初始化被跟蹤的User: " + entry.getName()); } cookie = retrieveCookie(ctx); ctx.setRequestControls(new Control[]{new DirSyncControl(1, Integer.MAX_VALUE, cookie, true)}); answer = ctx.search(searchBase, searchFilter, searchCtls); } logger.info("初始化總數:{};頁數:{}", totalResults, page); ctx.close(); ctx = new InitialLdapContext(env, null); //獲取到cookie後,一下是輪詢,每次輪詢完後更新cookie while (true) { TimeUnit.SECONDS.sleep(2); totalResults = 0; //每次輪詢都傳入上次更新過的cookie ctx.setRequestControls(new Control[]{new DirSyncControl(1, Integer.MAX_VALUE, cookie, true)}); answer = ctx.search(searchBase, searchFilter, searchCtls); while (answer.hasMoreElements()) { SearchResult sr = (SearchResult) answer.next(); totalResults++; //rDn logger.info("rDn:{} ", sr.getName()); final Attributes attrs = sr.getAttributes(); if (attrs != null) { //處理修改 for (NamingEnumeration<? extends Attribute> names = attrs.getAll(); names.hasMore(); ) { Attribute attr = names.next(); if (attr.getAll().hasMore()) { System.out.println(attr.getID() + " : " + attr.get()); } } } } logger.info("新修改:{} ", totalResults); //更新cookie cookie = retrieveCookie(ctx); } } /** * 取出cookie * * @param ctx * @return * @throws NamingException */ private static byte[] retrieveCookie(LdapContext ctx) throws NamingException { Control[] controls; if ((controls = ctx.getResponseControls()) != null) { for (int i = 0; i < controls.length; i++) { if (controls[i] instanceof DirSyncResponseControl) { DirSyncResponseControl dirSyncResponseControl = (DirSyncResponseControl) controls[i]; return dirSyncResponseControl.getCookie(); } } } return new byte[0]; } }