miodrag%netscape.com 276d1ca762 Updated ldapjdk 4.05
1999-11-12 06:34:14 +00:00

485 lines
15 KiB
Java

/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*-
*
* The contents of this file are subject to the Netscape Public
* License Version 1.1 (the "License"); you may not use this file
* except in compliance with the License. You may obtain a copy of
* the License at http://www.mozilla.org/NPL/
*
* Software distributed under the License is distributed on an "AS
* IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or
* implied. See the License for the specific language governing
* rights and limitations under the License.
*
* The Original Code is mozilla.org code.
*
* The Initial Developer of the Original Code is Netscape
* Communications Corporation. Portions created by Netscape are
* Copyright (C) 1999 Netscape Communications Corporation. All
* Rights Reserved.
*
* Contributor(s):
*/
package netscape.ldap.util;
import java.util.*;
import netscape.ldap.*;
/**
* Class to maintain a pool of individual connections to the
* same server. Specify the initial size and the max size
* when constructing a pool. Call getConnection() to obtain
* a connection from the pool and close() to return it. If
* the pool is fully extended and there are no free connections,
* getConnection() blocks until a connection has been returned
* to the pool.<BR>
* Call destroy() to release all connections.
*<BR><BR>Example:<BR>
*<PRE>
* ConnectionPool pool = null;
* try {
* pool = new ConnectionPool( 10, 30,
* "foo.acme.com",389,
* "uid=me, o=acme.com",
* "password" );
* } catch ( LDAPException e ) {
* System.err.println( "Unable to create connection pool" );
* System.exit( 1 );
* }
* while ( clientsKnocking ) {
* String filter = getSearchFilter();
* LDAPConnection ld = pool.getConnection();
* try {
* LDAPSearchResults res = ld.search( BASE, ld.SCOPE_SUB,
* filter, attrs,
* false );
* pool.close( ld );
* while( res.hasMoreElements() ) {
* ...
*</PRE>
*/
/**
* Connection pool, typically used by a server to avoid creating
* a new connection for each client
*
* @version 1.0
**/
public class ConnectionPool {
/**
* Constructor for specifying all parameters
*
* @param min initial number of connections
* @param max maximum number of connections
* @param host hostname of LDAP server
* @param port port number of LDAP server
* @param authdn DN to authenticate as
* @param authpw password for authentication
* @exception LDAPException on failure to create connections
*/
public ConnectionPool( int min, int max,
String host, int port,
String authdn, String authpw )
throws LDAPException {
this( min, max, host, port, authdn, authpw, null );
}
/**
* Constructor for specifying all parameters, anonymous
* identity
*
* @param min initial number of connections
* @param max maximum number of connections
* @param host hostname of LDAP server
* @param port port number of LDAP server
* @exception LDAPException on failure to create connections
*/
public ConnectionPool( int min, int max,
String host, int port)
throws LDAPException {
this( min, max, host, port, "", "");
}
/**
* Constructor for using default parameters, anonymous identity
*
* @param host hostname of LDAP server
* @param port port number of LDAP server
* @exception LDAPException on failure to create connections
*/
public ConnectionPool( String host, int port )
throws LDAPException {
// poolsize=10,max=20,host,port,
// noauth,nopswd
this( 10, 20, host, port, "", "" );
}
/*
* Constructor for using an existing connection to clone
* from
*
* @param min initial number of connections
* @param max maximum number of connections
* @param ldc connection to clone
* @exception LDAPException on failure to create connections
*/
public ConnectionPool( int min, int max, LDAPConnection ldc )
throws LDAPException {
this( min, max, ldc.getHost(), ldc.getPort(),
ldc.getAuthenticationDN(), ldc.getAuthenticationPassword(),
ldc );
}
/*
* Constructor for using an existing connection to clone
* from
*
* @param min initial number of connections
* @param max maximum number of connections
* @param host hostname of LDAP server
* @param port port number of LDAP server
* @param authdn DN to authenticate as
* @param authpw password for authentication
* @param ldc connection to clone
* @exception LDAPException on failure to create connections
*/
private ConnectionPool( int min, int max,
String host, int port,
String authdn, String authpw,
LDAPConnection ldc )
throws LDAPException {
this.poolSize = min;
this.poolMax = max;
this.host = host;
this.port = port;
this.authdn = authdn;
this.authpw = authpw;
this.ldc = ldc;
this.debugMode = false;
createPool();
}
/**
* Destroy the whole pool - called during a shutdown
*/
public void destroy() {
for ( int i = 0; i < pool.size(); i++ ) {
disconnect(
(LDAPConnectionObject)pool.elementAt(i) );
}
pool.removeAllElements();
}
/**
* Gets a connection from the pool
*
* If no connections are available, the pool will be
* extended if the number of connections is less than
* the maximum; if the pool cannot be extended, the method
* blocks until a free connection becomes available.
*
* @return an active connection.
*/
public LDAPConnection getConnection() {
LDAPConnection con;
while( (con = getConnFromPool()) == null ) {
synchronized( pool ) {
try {
pool.wait();
} catch ( InterruptedException e ) {
}
}
}
return con;
}
/**
* Gets a connection from the pool
*
* If no connections are available, the pool will be
* extended if the number of connections is less than
* the maximum; if the pool cannot be extended, the method
* returns null.
*
* @return an active connection or null.
*/
protected synchronized LDAPConnection getConnFromPool() {
LDAPConnection con = null;
LDAPConnectionObject ldapconnobj = null;
int pSize = pool.size();
// Get an available connection
for ( int i = 0; i < pSize; i++ ) {
// Get the ConnectionObject from the pool
LDAPConnectionObject co =
(LDAPConnectionObject)pool.elementAt(i);
if ( co.isAvailable() ) { // Conn available?
ldapconnobj = co;
break;
}
}
if ( ldapconnobj == null ) {
// If there there were no conns in pool, can we grow
// the pool?
if ( (poolMax < 0) ||
( (poolMax > 0) &&
(pSize < poolMax)) ) {
// Yes we can grow it
int i = addConnection();
// If a new connection was created, use it
if ( i >= 0 ) {
ldapconnobj =
(LDAPConnectionObject)pool.elementAt(i);
}
} else {
debug("All pool connections in use",true);
}
}
if ( ldapconnobj != null ) {
ldapconnobj.setInUse( true ); // Mark as in use
con = ldapconnobj.getLDAPConn();
}
return con;
}
/**
* This is our soft close - all we do is mark
* the connection as available for others to use.
*
* @param ld a connection to return to the pool
*/
public synchronized void close( LDAPConnection ld ) {
int index = find( ld );
if ( index != -1 ) {
LDAPConnectionObject co =
(LDAPConnectionObject)pool.elementAt(index);
co.setInUse( false ); // Mark as available
synchronized( pool ) {
pool.notifyAll();
}
}
}
/**
* Debug method to print the contents of the pool
*/
public void printPool(){
System.out.println("--ConnectionPool--");
for ( int i = 0; i < pool.size(); i++ ) {
LDAPConnectionObject co =
(LDAPConnectionObject)pool.elementAt(i);
System.out.println( "" + i + "=" + co );
}
}
private void disconnect(
LDAPConnectionObject ldapconnObject ) {
if ( ldapconnObject != null ) {
if (ldapconnObject.isAvailable()) {
LDAPConnection ld = ldapconnObject.getLDAPConn();
if ( (ld != null) && (ld.isConnected()) ) {
try {
ld.disconnect();
} catch (LDAPException e) {
debug("disconnect: "+e.toString());
}
}
ldapconnObject.setLDAPConn(null); // Clear conn
}
}
}
private void createPool() throws LDAPException {
// Called by the constructors
if ( poolSize <= 0 ) {
throw new LDAPException("ConnectionPoolSize invalid");
}
if ( poolMax < poolSize ) {
debug("ConnectionPoolMax is invalid, set to " +
poolSize);
poolMax = poolSize;
}
debug("****Initializing LDAP Pool****",true);
debug("LDAP host = "+host+" on port "+port,true);
debug("Number of connections="+poolSize,true);
debug("Maximum number of connections="+poolMax,true);
debug("******",true);
pool = new java.util.Vector(); // Create pool vector
setUpPool( poolSize ); // Initialize it
}
private int addConnection() {
int index = -1;
debug("adding a connection to pool...");
try {
int size = pool.size() + 1; // Add one connection
setUpPool( size );
if ( size == pool.size() ) {
// New size is size requested?
index = size - 1;
}
} catch (Exception ex) {
debug("Adding a connection: "+ex.toString(),true);
}
return index;
}
private synchronized void setUpPool( int size )
throws LDAPException {
// Loop on creating connections
while( pool.size() < size ) {
LDAPConnectionObject co =
new LDAPConnectionObject();
// Make LDAP connection, using template if available
LDAPConnection newConn =
(ldc != null) ? (LDAPConnection)ldc.clone() :
new LDAPConnection();
co.setLDAPConn(newConn);
try {
if ( newConn.isConnected() ) {
// If using a template, then reconnect
// to create a separate physical connection
newConn.reconnect();
} else {
// Not using a template, so connect with
// simple authentication
newConn.connect( host, port,
authdn, authpw);
}
} catch ( LDAPException le ) {
debug("Creating pool:"+le.toString(),true);
debug("aborting....",true);
throw le;
}
co.setInUse( false ); // Mark not in use
pool.addElement( co );
}
}
private int find( LDAPConnection con ) {
// Find the matching Connection in the pool
if ( con != null ) {
for ( int i = 0; i < pool.size(); i++ ) {
LDAPConnectionObject co =
(LDAPConnectionObject)pool.elementAt(i);
if ( co.getLDAPConn() == con ) {
return i;
}
}
}
return -1;
}
/**
* Sets the debug printout mode.
*
* @param mode debug mode to use
*/
public synchronized void setDebug( boolean mode ) {
debugMode = mode;
}
/**
* Reports the debug printout mode.
*
* @return debug mode in use.
*/
public boolean getDebug() {
return debugMode;
}
private void debug( String s ) {
if ( debugMode )
System.out.println("ConnectionPool ("+
new Date()+") : " + s);
}
private void debug(String s, boolean severe) {
if ( debugMode || severe ) {
System.out.println("ConnectionPool ("+
new Date()+") : " + s);
}
}
/**
* Wrapper for LDAPConnection object in pool
*/
class LDAPConnectionObject{
/**
* Returns the associated LDAPConnection.
*
* @return the LDAPConnection.
*
*/
LDAPConnection getLDAPConn() {
return this.ld;
}
/**
* Sets the associated LDAPConnection
*
* @param ld the LDAPConnection
*
*/
void setLDAPConn( LDAPConnection ld ) {
this.ld = ld;
}
/**
* Marks a connection in use or available
*
* @param inUse <code>true</code> to mark in use, <code>false</code> if available
*
*/
void setInUse( boolean inUse ) {
this.inUse = inUse;
}
/**
* Returns whether the connection is available
* for use by another user.
*
* @return <code>true</code> if available.
*/
boolean isAvailable() {
return !inUse;
}
/**
* Debug method
*
* @return s user-friendly rendering of the object.
*/
public String toString() {
return "LDAPConnection=" + ld + ",inUse=" + inUse;
}
private LDAPConnection ld; // LDAP Connection
private boolean inUse; // In use? (true = yes)
}
private int poolSize; // Min pool size
private int poolMax; // Max pool size
private String host; // LDAP host
private int port; // Port to connect at
private String authdn; // Identity of connections
private String authpw; // Password for authdn
private LDAPConnection ldc = null; // Connection to clone
private java.util.Vector pool; // the actual pool
private boolean debugMode;
}