LDAP Authentication using Java

前端 未结 4 1424
面向向阳花
面向向阳花 2020-12-12 15:23

I need to do LDAP Authentication for an application.

I tried the following program:

import java.util.Hashtable;  

import javax.naming.Context;  
imp         


        
4条回答
  •  时光取名叫无心
    2020-12-12 15:51

    This is my LDAP Java login test application supporting LDAP:// and LDAPS:// self-signed test certificate. Code is taken from few SO posts, simplified implementation and removed legacy sun.java.* imports.

    • how to accept self-signed certificates for JNDI/LDAP connections?
    • Authenticating against Active Directory with Java on Linux

    Usage
    I have run this in Windows7 and Linux machines against WinAD directory service. Application prints username and member groups.

    $ java -cp classes test.LoginLDAP url=ldap://1.2.3.4:389 username=myname@company.fi password=mypwd

    $ java -cp classes test.LoginLDAP url=ldaps://1.2.3.4:636 username=myname@company.fi password=mypwd

    Test application supports temporary self-signed test certificates for ldaps:// protocol, this DummySSLFactory accepts any server cert so man-in-the-middle is possible. Real life installation should import server certificate to a local JKS keystore file and not using dummy factory.

    Application uses enduser's username+password for initial context and ldap queries, it works for WinAD but don't know if can be used for all ldap server implementations. You could create context with internal username+pwd then run queries to see if given enduser is found.

    LoginLDAP.java

    package test;
    
    import java.util.*;
    import javax.naming.*;
    import javax.naming.directory.*;
    
    public class LoginLDAP {
    
        public static void main(String[] args) throws Exception {
            Map params = createParams(args);
    
            String url = params.get("url"); // ldap://1.2.3.4:389 or ldaps://1.2.3.4:636
            String principalName = params.get("username"); // firstname.lastname@mydomain.com
            String domainName = params.get("domain"); // mydomain.com or empty
    
            if (domainName==null || "".equals(domainName)) {
                int delim = principalName.indexOf('@');
                domainName = principalName.substring(delim+1);
            }
    
            Properties props = new Properties();
            props.put(Context.INITIAL_CONTEXT_FACTORY, "com.sun.jndi.ldap.LdapCtxFactory");
            props.put(Context.PROVIDER_URL, url); 
            props.put(Context.SECURITY_PRINCIPAL, principalName); 
            props.put(Context.SECURITY_CREDENTIALS, params.get("password")); // secretpwd
            if (url.toUpperCase().startsWith("LDAPS://")) {
                props.put(Context.SECURITY_PROTOCOL, "ssl");
                props.put(Context.SECURITY_AUTHENTICATION, "simple");
                props.put("java.naming.ldap.factory.socket", "test.DummySSLSocketFactory");         
            }
    
            InitialDirContext context = new InitialDirContext(props);
            try {
                SearchControls ctrls = new SearchControls();
                ctrls.setSearchScope(SearchControls.SUBTREE_SCOPE);
                NamingEnumeration results = context.search(toDC(domainName),"(& (userPrincipalName="+principalName+")(objectClass=user))", ctrls);
                if(!results.hasMore())
                    throw new AuthenticationException("Principal name not found");
    
                SearchResult result = results.next();
                System.out.println("distinguisedName: " + result.getNameInNamespace() ); // CN=Firstname Lastname,OU=Mycity,DC=mydomain,DC=com
    
                Attribute memberOf = result.getAttributes().get("memberOf");
                if(memberOf!=null) {
                    for(int idx=0; idx0)  buf.append(",");
                buf.append("DC=").append(token);
            }
            return buf.toString();
        }
    
        private static Map createParams(String[] args) {
            Map params = new HashMap();  
            for(String str : args) {
                int delim = str.indexOf('=');
                if (delim>0) params.put(str.substring(0, delim).trim(), str.substring(delim+1).trim());
                else if (delim==0) params.put("", str.substring(1).trim());
                else params.put(str, null);
            }
            return params;
        }
    
    }
    

    And SSL helper class.

    package test;
    
    import java.io.*;
    import java.net.*;
    import java.security.SecureRandom;
    import java.security.cert.X509Certificate;    
    import javax.net.*;
    import javax.net.ssl.*;
    
    public class DummySSLSocketFactory extends SSLSocketFactory {
        private SSLSocketFactory socketFactory;
        public DummySSLSocketFactory() {
            try {
              SSLContext ctx = SSLContext.getInstance("TLS");
              ctx.init(null, new TrustManager[]{ new DummyTrustManager()}, new SecureRandom());
              socketFactory = ctx.getSocketFactory();
            } catch ( Exception ex ){ throw new IllegalArgumentException(ex); }
        }
    
          public static SocketFactory getDefault() { return new DummySSLSocketFactory(); }
    
          @Override public String[] getDefaultCipherSuites() { return socketFactory.getDefaultCipherSuites(); }
          @Override public String[] getSupportedCipherSuites() { return socketFactory.getSupportedCipherSuites(); }
    
          @Override public Socket createSocket(Socket socket, String string, int i, boolean bln) throws IOException {
            return socketFactory.createSocket(socket, string, i, bln);
          }
          @Override public Socket createSocket(String string, int i) throws IOException, UnknownHostException {
            return socketFactory.createSocket(string, i);
          }
          @Override public Socket createSocket(String string, int i, InetAddress ia, int i1) throws IOException, UnknownHostException {
            return socketFactory.createSocket(string, i, ia, i1);
          }
          @Override public Socket createSocket(InetAddress ia, int i) throws IOException {
            return socketFactory.createSocket(ia, i);
          }
          @Override public Socket createSocket(InetAddress ia, int i, InetAddress ia1, int i1) throws IOException {
            return socketFactory.createSocket(ia, i, ia1, i1);
          }
    }
    
    class DummyTrustManager implements X509TrustManager {
        @Override public void checkClientTrusted(X509Certificate[] xcs, String str) {
            // do nothing
        }
        @Override public void checkServerTrusted(X509Certificate[] xcs, String str) {
            /*System.out.println("checkServerTrusted for authType: " + str); // RSA
            for(int idx=0; idx

提交回复
热议问题