org.jboss.naming.remote.client.NamingStoreCache Maven / Gradle / Ivy
Go to download
This artifact provides a single jar that contains all classes required to use remote Jakarta Enterprise Beans and Jakarta Messaging, including
all dependencies. It is intended for use by those not using maven, maven users should just import the Jakarta Enterprise Beans and
Jakarta Messaging BOM's instead (shaded JAR's cause lots of problems with maven, as it is very easy to inadvertently end up
with different versions on classes on the class path).
package org.jboss.naming.remote.client;
import java.io.Closeable;
import java.io.IOException;
import java.net.URI;
import java.net.URISyntaxException;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;
import java.util.concurrent.atomic.AtomicInteger;
import javax.naming.NamingException;
import javax.security.auth.callback.CallbackHandler;
import org.jboss.logging.Logger;
import org.jboss.naming.remote.client.ejb.EJBClientHandler;
import org.jboss.remoting3.Channel;
import org.jboss.remoting3.Endpoint;
import org.xnio.OptionMap;
/**
* @author John Bailey
* @author Stuart Douglas
*/
public class NamingStoreCache {
private static final Logger logger = Logger.getLogger(NamingStoreCache.class);
private final ConcurrentMap cache = new ConcurrentHashMap();
/**
* Returns a {@link Channel} for the passed connection properties. If the connection is already created
* and cached for the passed connection properties, then the cached channel will be returned. Else a new
* connection and channel will be created and that new channel returned.
*
* @param clientEndpoint The {@link org.jboss.remoting3.Endpoint} that will be used to open a connection
* @param connectionURL The connection URL
* @param connectOptions The options to be used for connection creation
* @param callbackHandler The callback handler to be used for connection creation
* @param connectionTimeout The connection timeout in milli seconds that will be used while creating a connection
* @param channelCreationOptions The {@link org.xnio.OptionMap options} that will be used if/when the channel is created
* @param channelCreationTimeoutInMillis The timeout in milli seconds, that will be used while opening a channel
* @param contextCloseTasks The tasks to be performed when the context is closed
* @return
* @throws IOException
*/
public synchronized RemoteNamingStore getRemoteNamingStore(final Endpoint clientEndpoint, final String connectionURL, final OptionMap connectOptions, final CallbackHandler callbackHandler, final long connectionTimeout,
final OptionMap channelCreationOptions, final long channelCreationTimeoutInMillis, final List contextCloseTasks, boolean randomServer) throws IOException, NamingException, URISyntaxException {
return getRemoteNamingStore(clientEndpoint, connectionURL, connectOptions, callbackHandler, connectionTimeout, channelCreationOptions, channelCreationTimeoutInMillis, contextCloseTasks, randomServer, null);
}
/**
* Returns a {@link Channel} for the passed connection properties. If the connection is already created
* and cached for the passed connection properties, then the cached channel will be returned. Else a new
* connection and channel will be created and that new channel returned.
*
* @param clientEndpoint The {@link org.jboss.remoting3.Endpoint} that will be used to open a connection
* @param connectionURL The connection URL
* @param connectOptions The options to be used for connection creation
* @param callbackHandler The callback handler to be used for connection creation
* @param connectionTimeout The connection timeout in milli seconds that will be used while creating a connection
* @param channelCreationOptions The {@link org.xnio.OptionMap options} that will be used if/when the channel is created
* @param channelCreationTimeoutInMillis The timeout in milli seconds, that will be used while opening a channel
* @param contextCloseTasks The tasks to be performed when the context is closed
* @return
* @throws IOException
*/
public synchronized RemoteNamingStore getRemoteNamingStore(final Endpoint clientEndpoint, final String connectionURL, final OptionMap connectOptions, final CallbackHandler callbackHandler, final long connectionTimeout,
final OptionMap channelCreationOptions, final long channelCreationTimeoutInMillis, final List contextCloseTasks, boolean randomServer,
final EJBClientHandler ejbClientHandler) throws IOException, NamingException, URISyntaxException {
final CacheKey key = new CacheKey(clientEndpoint, callbackHandler, connectOptions, connectionURL, ejbClientHandler);
CacheEntry cacheEntry = cache.get(key);
if (cacheEntry == null) {
RemoteNamingStore store;
if (connectionURL.contains(",")) {
//HA context
String[] urls = connectionURL.split(",");
List connectionUris = new ArrayList(urls.length);
for (final String url : urls) {
connectionUris.add(new URI(url.trim()));
}
store = new HaRemoteNamingStore(channelCreationTimeoutInMillis, channelCreationOptions, connectionTimeout, callbackHandler, connectOptions, connectionUris, clientEndpoint, randomServer, ejbClientHandler);
} else {
store = new HaRemoteNamingStore(channelCreationTimeoutInMillis, channelCreationOptions, connectionTimeout,
callbackHandler, connectOptions, Collections.singletonList(new URI(connectionURL.trim())),
clientEndpoint, randomServer, ejbClientHandler);
}
cacheEntry = new CacheEntry(store);
cache.put(key, cacheEntry);
}
//when the context is closed we need to release and decrease the reference count
contextCloseTasks.add(new RemoteContext.CloseTask() {
@Override
public void close(final boolean isFinalize) {
release(key, isFinalize);
}
});
cacheEntry.referenceCount.incrementAndGet();
return cacheEntry.namingStore;
}
public synchronized void release(final CacheKey connectionHash, final boolean async) {
final CacheEntry cacheEntry = cache.get(connectionHash);
if (cacheEntry == null) {
return;
}
if (cacheEntry.referenceCount.decrementAndGet() == 0) {
try {
if (async) {
cacheEntry.namingStore.closeAsync();
} else {
cacheEntry.namingStore.close();
}
} catch (NamingException e) {
throw new RuntimeException("Failed to close naming store", e);
} finally {
cache.remove(connectionHash);
}
}
}
public synchronized void shutdown() {
for (Map.Entry entry : cache.entrySet()) {
final RemoteNamingStore namingStore = entry.getValue().namingStore;
try {
namingStore.close();
} catch (Throwable t) {
logger.debug("Failed to close naming store ", t);
}
}
}
private class CacheEntry {
private final AtomicInteger referenceCount = new AtomicInteger(0);
private final RemoteNamingStore namingStore;
private CacheEntry(final RemoteNamingStore namingStore) {
this.namingStore = namingStore;
}
}
private static final class CacheKey {
final Endpoint endpoint;
final String destination;
final OptionMap connectOptions;
final CallbackHandler callbackHandler;
final EJBClientHandler ejbClientHandler;
private CacheKey(final Endpoint endpoint, final CallbackHandler callbackHandler, final OptionMap connectOptions, final String destination, final EJBClientHandler ejbClientHandler) {
this.endpoint = endpoint;
this.callbackHandler = callbackHandler;
this.connectOptions = connectOptions;
this.destination = destination;
this.ejbClientHandler = ejbClientHandler;
}
@Override
public boolean equals(Object o) {
if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false;
CacheKey cacheKey = (CacheKey) o;
if (callbackHandler != null ? !callbackHandler.equals(cacheKey.callbackHandler) : cacheKey.callbackHandler != null)
return false;
if (connectOptions != null ? !connectOptions.equals(cacheKey.connectOptions) : cacheKey.connectOptions != null)
return false;
if (destination != null ? !destination.equals(cacheKey.destination) : cacheKey.destination != null)
return false;
if (endpoint != null ? !endpoint.equals(cacheKey.endpoint) : cacheKey.endpoint != null) return false;
if (ejbClientHandler != null ? !ejbClientHandler.equals(cacheKey.ejbClientHandler) : cacheKey.ejbClientHandler != null) return false;
return true;
}
@Override
public int hashCode() {
int result = endpoint != null ? endpoint.hashCode() : 0;
result = 31 * result + (destination != null ? destination.hashCode() : 0);
result = 31 * result + (connectOptions != null ? connectOptions.hashCode() : 0);
result = 31 * result + (callbackHandler != null ? callbackHandler.hashCode() : 0);
result = 31 * result + (ejbClientHandler != null ? ejbClientHandler.hashCode() : 0);
return result;
}
}
private static void safeClose(Closeable closable) {
try {
closable.close();
} catch (Throwable t) {
logger.debug("Failed to close connection ", t);
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy