|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
*/ |
|
|
|
package com.sun.jndi.ldap.pool; |
|
|
|
import java.util.ArrayList; |
|
import java.util.List; |
|
|
|
import java.lang.ref.Reference; |
|
import java.lang.ref.SoftReference; |
|
|
|
import javax.naming.NamingException; |
|
import javax.naming.InterruptedNamingException; |
|
import javax.naming.CommunicationException; |
|
|
|
/** |
|
* Represents a list of PooledConnections (actually, ConnectionDescs) with the |
|
* same pool id. |
|
* The list starts out with an initial number of connections. |
|
* Additional PooledConnections are created lazily upon demand. |
|
* The list has a maximum size. When the number of connections |
|
* reaches the maximum size, a request for a PooledConnection blocks until |
|
* a connection is returned to the list. A maximum size of zero means that |
|
* there is no maximum: connection creation will be attempted when |
|
* no idle connection is available. |
|
* |
|
* The list may also have a preferred size. If the current list size |
|
* is less than the preferred size, a request for a connection will result in |
|
* a PooledConnection being created (even if an idle connection is available). |
|
* If the current list size is greater than the preferred size, |
|
* a connection being returned to the list will be closed and removed from |
|
* the list. A preferred size of zero means that there is no preferred size: |
|
* connections are created only when no idle connection is available and |
|
* a connection being returned to the list is not closed. Regardless of the |
|
* preferred size, connection creation always observes the maximum size: |
|
* a connection won't be created if the list size is at or exceeds the |
|
* maximum size. |
|
* |
|
* @author Rosanna Lee |
|
*/ |
|
|
|
|
|
final class Connections implements PoolCallback { |
|
private static final boolean debug = Pool.debug; |
|
private static final boolean trace = |
|
com.sun.jndi.ldap.LdapPoolManager.trace; |
|
private static final int DEFAULT_SIZE = 10; |
|
|
|
final private int maxSize; |
|
final private int prefSize; |
|
final private List<ConnectionDesc> conns; |
|
|
|
private boolean closed = false; |
|
private Reference<Object> ref; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
*/ |
|
Connections(Object id, int initSize, int prefSize, int maxSize, |
|
PooledConnectionFactory factory) throws NamingException { |
|
|
|
this.maxSize = maxSize; |
|
if (maxSize > 0) { |
|
|
|
this.prefSize = Math.min(prefSize, maxSize); |
|
initSize = Math.min(initSize, maxSize); |
|
} else { |
|
this.prefSize = prefSize; |
|
} |
|
conns = new ArrayList<>(maxSize > 0 ? maxSize : DEFAULT_SIZE); |
|
|
|
// Maintain soft ref to id so that this Connections' entry in |
|
|
|
ref = new SoftReference<>(id); |
|
|
|
d("init size=", initSize); |
|
d("max size=", maxSize); |
|
d("preferred size=", prefSize); |
|
|
|
|
|
PooledConnection conn; |
|
for (int i = 0; i < initSize; i++) { |
|
conn = factory.createPooledConnection(this); |
|
td("Create ", conn ,factory); |
|
conns.add(new ConnectionDesc(conn)); |
|
} |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
*/ |
|
synchronized PooledConnection get(long timeout, |
|
PooledConnectionFactory factory) throws NamingException { |
|
PooledConnection conn; |
|
long start = (timeout > 0 ? System.currentTimeMillis() : 0); |
|
long waittime = timeout; |
|
|
|
d("get(): before"); |
|
while ((conn = getOrCreateConnection(factory)) == null) { |
|
if (timeout > 0 && waittime <= 0) { |
|
throw new CommunicationException( |
|
"Timeout exceeded while waiting for a connection: " + |
|
timeout + "ms"); |
|
} |
|
try { |
|
d("get(): waiting"); |
|
if (waittime > 0) { |
|
wait(waittime); |
|
} else { |
|
wait(); |
|
} |
|
} catch (InterruptedException e) { |
|
throw new InterruptedNamingException( |
|
"Interrupted while waiting for a connection"); |
|
} |
|
|
|
if (timeout > 0) { |
|
long now = System.currentTimeMillis(); |
|
waittime = timeout - (now - start); |
|
} |
|
} |
|
|
|
d("get(): after"); |
|
return conn; |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
*/ |
|
private PooledConnection getOrCreateConnection( |
|
PooledConnectionFactory factory) throws NamingException { |
|
|
|
int size = conns.size(); |
|
PooledConnection conn = null; |
|
|
|
if (prefSize <= 0 || size >= prefSize) { |
|
// If no prefSize specified, or list size already meets or |
|
|
|
ConnectionDesc entry; |
|
for (int i = 0; i < size; i++) { |
|
entry = conns.get(i); |
|
if ((conn = entry.tryUse()) != null) { |
|
d("get(): use ", conn); |
|
td("Use ", conn); |
|
return conn; |
|
} |
|
} |
|
} |
|
|
|
|
|
if (maxSize > 0 && size >= maxSize) { |
|
return null; |
|
} |
|
|
|
conn = factory.createPooledConnection(this); |
|
td("Create and use ", conn, factory); |
|
conns.add(new ConnectionDesc(conn, true)); |
|
|
|
return conn; |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
*/ |
|
public synchronized boolean releasePooledConnection(PooledConnection conn) { |
|
ConnectionDesc entry; |
|
int loc = conns.indexOf(entry=new ConnectionDesc(conn)); |
|
|
|
d("release(): ", conn); |
|
|
|
if (loc >= 0) { |
|
// Found entry |
|
|
|
if (closed || (prefSize > 0 && conns.size() > prefSize)) { |
|
// If list size exceeds prefSize, close connection |
|
|
|
d("release(): closing ", conn); |
|
td("Close ", conn); |
|
|
|
|
|
conns.remove(entry); |
|
conn.closeConnection(); |
|
|
|
} else { |
|
d("release(): release ", conn); |
|
td("Release ", conn); |
|
|
|
|
|
entry = conns.get(loc); |
|
|
|
entry.release(); |
|
} |
|
notifyAll(); |
|
d("release(): notify"); |
|
return true; |
|
} else { |
|
return false; |
|
} |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
*/ |
|
public synchronized boolean removePooledConnection(PooledConnection conn) { |
|
if (conns.remove(new ConnectionDesc(conn))) { |
|
d("remove(): ", conn); |
|
|
|
notifyAll(); |
|
|
|
d("remove(): notify"); |
|
td("Remove ", conn); |
|
|
|
if (conns.isEmpty()) { |
|
// Remove softref to make pool entry eligible for GC. |
|
|
|
ref = null; |
|
} |
|
|
|
return true; |
|
} else { |
|
d("remove(): not found ", conn); |
|
return false; |
|
} |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
*/ |
|
boolean expire(long threshold) { |
|
List<ConnectionDesc> clonedConns; |
|
synchronized(this) { |
|
clonedConns = new ArrayList<>(conns); |
|
} |
|
List<ConnectionDesc> expired = new ArrayList<>(); |
|
|
|
for (ConnectionDesc entry : clonedConns) { |
|
d("expire(): ", entry); |
|
if (entry.expire(threshold)) { |
|
expired.add(entry); |
|
td("expire(): Expired ", entry); |
|
} |
|
} |
|
|
|
synchronized (this) { |
|
conns.removeAll(expired); |
|
// Don't need to call notify() because we're |
|
// removing only idle connections. If there were |
|
// idle connections, then there should be no waiters. |
|
return conns.isEmpty(); |
|
} |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
*/ |
|
synchronized void close() { |
|
expire(System.currentTimeMillis()); |
|
closed = true; |
|
} |
|
|
|
String getStats() { |
|
int idle = 0; |
|
int busy = 0; |
|
int expired = 0; |
|
long use = 0; |
|
int len; |
|
|
|
synchronized (this) { |
|
len = conns.size(); |
|
|
|
ConnectionDesc entry; |
|
for (int i = 0; i < len; i++) { |
|
entry = conns.get(i); |
|
use += entry.getUseCount(); |
|
switch (entry.getState()) { |
|
case ConnectionDesc.BUSY: |
|
++busy; |
|
break; |
|
case ConnectionDesc.IDLE: |
|
++idle; |
|
break; |
|
case ConnectionDesc.EXPIRED: |
|
++expired; |
|
} |
|
} |
|
} |
|
return "size=" + len + "; use=" + use + "; busy=" + busy |
|
+ "; idle=" + idle + "; expired=" + expired; |
|
} |
|
|
|
private void d(String msg, Object o1) { |
|
if (debug) { |
|
d(msg + o1); |
|
} |
|
} |
|
|
|
private void d(String msg, int i) { |
|
if (debug) { |
|
d(msg + i); |
|
} |
|
} |
|
|
|
private void d(String msg) { |
|
if (debug) { |
|
System.err.println(this + "." + msg + "; size: " + conns.size()); |
|
} |
|
} |
|
|
|
private void td(String msg, Object o1, Object o2) { |
|
if (trace) { |
|
td(msg + o1 + "[" + o2 + "]"); |
|
} |
|
} |
|
private void td(String msg, Object o1) { |
|
if (trace) { |
|
td(msg + o1); |
|
} |
|
} |
|
private void td(String msg) { |
|
if (trace) { |
|
System.err.println(msg); |
|
} |
|
} |
|
} |