org.apache.openejb.resource.activemq.ActiveMQ5Factory Maven / Gradle / Ivy
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You under the Apache License, Version 2.0
* (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.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.apache.openejb.resource.activemq;
import org.apache.activemq.broker.BrokerFactory;
import org.apache.activemq.broker.BrokerFactoryHandler;
import org.apache.activemq.broker.BrokerPlugin;
import org.apache.activemq.broker.BrokerService;
import org.apache.activemq.broker.jmx.ManagementContext;
import org.apache.activemq.network.DiscoveryNetworkConnector;
import org.apache.activemq.network.NetworkConnector;
import org.apache.activemq.ra.ActiveMQResourceAdapter;
import org.apache.activemq.store.PersistenceAdapter;
import org.apache.activemq.store.PersistenceAdapterFactory;
import org.apache.activemq.store.jdbc.JDBCPersistenceAdapter;
import org.apache.activemq.store.memory.MemoryPersistenceAdapter;
import org.apache.activemq.usage.SystemUsage;
import org.apache.activemq.util.IntrospectionSupport;
import org.apache.activemq.util.URISupport;
import org.apache.openejb.loader.SystemInstance;
import org.apache.openejb.spi.ContainerSystem;
import org.apache.openejb.util.LogCategory;
import org.apache.openejb.util.Logger;
import org.apache.xbean.propertyeditor.PropertyEditorException;
import org.apache.xbean.propertyeditor.PropertyEditorRegistry;
import org.apache.xbean.recipe.ObjectRecipe;
import javax.naming.Context;
import javax.naming.NamingException;
import javax.sql.DataSource;
import java.io.IOException;
import java.io.UnsupportedEncodingException;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.net.URI;
import java.net.URISyntaxException;
import java.net.URLEncoder;
import java.util.Collection;
import java.util.HashMap;
import java.util.Iterator;
import java.util.LinkedList;
import java.util.Locale;
import java.util.Map;
import java.util.Properties;
import java.util.concurrent.atomic.AtomicBoolean;
public class ActiveMQ5Factory implements BrokerFactoryHandler {
private static Properties properties;
protected static final Map brokers = new HashMap();
private static Throwable throwable;
private static final AtomicBoolean started = new AtomicBoolean(false);
private static PropertyEditorRegistry propertyEditorRegistry = new PropertyEditorRegistry().registerDefaults();
public static void setThreadProperties(final Properties p) {
properties = p;
}
@Override
public synchronized BrokerService createBroker(final URI brokerURI) throws Exception {
Logger.getInstance(LogCategory.OPENEJB_STARTUP, ActiveMQ5Factory.class).getChildLogger("service").info("ActiveMQ5Factory creating broker");
BrokerService broker = brokers.get(brokerURI);
if (null == broker || !broker.isStarted()) {
final Properties properties = getLowerCaseProperties();
final URISupport.CompositeData compositeData = URISupport.parseComposite(new URI(brokerURI.getRawSchemeSpecificPart()));
final Map params = new HashMap<>(compositeData.getParameters());
final PersistenceAdapter persistenceAdapter;
if ("true".equals(params.remove("usekahadb"))) {
persistenceAdapter = createPersistenceAdapter("org.apache.activemq.store.kahadb.KahaDBPersistenceAdapter", "kahadb", params);
} else if ("true".equals(params.remove("useleveldb"))) {
persistenceAdapter = createPersistenceAdapter("org.apache.activemq.store.leveldb.LevelDBPersistenceAdapter", "leveldb", params);
} else if (params.get("persistenceadapter") != null) {
final String adapter = params.remove("persistenceadapter");
persistenceAdapter = createPersistenceAdapter(adapter, "persistence", params);
} else {
persistenceAdapter = null;
}
final String systemUsage = params.remove("systemUsage");
final SystemUsage systemUsageInstance;
if ("true".equalsIgnoreCase(systemUsage)) {
systemUsageInstance = newSystemUsage(params);
} else {
systemUsageInstance = null;
}
final BrokerPlugin[] plugins = createPlugins(params);
final URI uri = new URI(cleanUpUri(brokerURI.getRawSchemeSpecificPart(), compositeData.getParameters(), params));
broker = "broker".equals(uri.getScheme()) ? newDefaultBroker(uri) : BrokerFactory.createBroker(uri);
if (plugins != null) {
broker.setPlugins(plugins);
}
if (systemUsageInstance != null) {
broker.setSystemUsage(systemUsageInstance);
}
brokers.put(brokerURI, broker);
if (persistenceAdapter != null) {
broker.setPersistenceAdapter(persistenceAdapter);
// if user didn't set persistent to true then setPersistenceAdapter() alone is ignored so forcing it with the factory
broker.setPersistenceFactory(new ProvidedPersistenceAdapterPersistenceAdapterFactory(persistenceAdapter));
broker.setPersistent(true);
tomeeConfig(broker);
} else {
final boolean notXbean = !uri.getScheme().toLowerCase(Locale.ENGLISH).startsWith("xbean");
if (notXbean) {
Object value = properties.get("datasource");
if (String.class.isInstance(value) && value.toString().length() == 0) {
value = null;
}
final DataSource dataSource;
if (value != null) {
if (DataSource.class.isInstance(value)) {
dataSource = DataSource.class.cast(value);
} else if (String.class.isInstance(value)) {
final String resouceId = String.class.cast(value);
try {
final ContainerSystem containerSystem = SystemInstance.get().getComponent(ContainerSystem.class);
final Context context = containerSystem.getJNDIContext();
final Object obj = context.lookup("openejb/Resource/" + resouceId);
if (!(obj instanceof DataSource)) {
throw new IllegalArgumentException("Resource with id " + resouceId
+ " is not a DataSource, but is " + obj.getClass().getName());
}
dataSource = (DataSource) obj;
} catch (final NamingException e) {
throw new IllegalArgumentException("Unknown datasource " + resouceId);
}
} else {
throw new IllegalArgumentException("Unexpected datasource definition: " + value);
}
} else {
dataSource = null;
}
if (null != dataSource) {
final JDBCPersistenceAdapter adapter = new JDBCPersistenceAdapter();
if (properties.containsKey("usedatabaselock")) {
//This must be false for hsqldb
adapter.setUseLock(Boolean.parseBoolean(properties.getProperty("usedatabaselock", "true")));
}
adapter.setDataSource(dataSource);
broker.setPersistent(true);
broker.setPersistenceAdapter(adapter);
} else {
broker.setPersistenceAdapter(new MemoryPersistenceAdapter());
}
tomeeConfig(broker);
}
}
//We must close the broker
broker.setUseShutdownHook(false);
broker.setSystemExitOnShutdown(false);
broker.setStartAsync(false);
final ManagementContext managementContext = new ManagementContext();
managementContext.setCreateConnector(false);
broker.setManagementContext(managementContext);
final BrokerService bs = broker;
final Thread start = new Thread("ActiveMQFactory start and checkpoint") {
@Override
public void run() {
Thread.currentThread().setContextClassLoader(ActiveMQResourceAdapter.class.getClassLoader());
try {
//Start before returning - this is known to be safe.
if (!bs.isStarted()) {
Logger
.getInstance(LogCategory.OPENEJB_STARTUP, ActiveMQ5Factory.class)
.getChildLogger("service")
.info("Starting ActiveMQ BrokerService");
bs.start();
}
bs.waitUntilStarted();
//Force a checkpoint to initialize pools
Logger
.getInstance(LogCategory.OPENEJB_STARTUP, ActiveMQ5Factory.class)
.getChildLogger("service")
.info("Starting ActiveMQ checkpoint");
bs.getPersistenceAdapter().checkpoint(true);
started.set(true);
} catch (final Throwable t) {
throwable = t;
}
}
};
/*
* An application may require immediate access to JMS. So we need to block here until the service
* has started. How long ActiveMQ requires to actually create a broker is unpredictable.
*
* A broker in OpenEJB is usually a wrapper for an embedded ActiveMQ server service. The broker configuration
* allows the definition of a remote ActiveMQ server, in which case startup is not an issue as the broker is
* basically a client.
*
* If the broker is local and the message store contains millions of messages then the startup time is obviously going to
* be longer as these need to be indexed by ActiveMQ.
*
* A balanced timeout will always be use case dependent.
*/
int timeout = 30000;
try {
timeout = Integer.parseInt(properties.getProperty("startuptimeout", "30000"));
Logger
.getInstance(LogCategory.OPENEJB_STARTUP, ActiveMQ5Factory.class)
.getChildLogger("service")
.info("Using ActiveMQ startup timeout of " + timeout + "ms");
} catch (final Throwable e) {
//Ignore
}
start.setDaemon(true);
start.start();
try {
start.join(timeout);
} catch (final InterruptedException e) {
//Ignore
}
if (null != throwable) {
Logger.getInstance(LogCategory.OPENEJB_STARTUP, ActiveMQ5Factory.class).getChildLogger("service").error("ActiveMQ failed to start broker",
throwable);
} else if (started.get()) {
Logger.getInstance(LogCategory.OPENEJB_STARTUP, ActiveMQ5Factory.class).getChildLogger("service").info("ActiveMQ broker started");
} else {
Logger
.getInstance(LogCategory.OPENEJB_STARTUP, ActiveMQ5Factory.class)
.getChildLogger("service")
.warning("ActiveMQ failed to start broker within " + timeout + " seconds - It may be unusable");
}
}
return broker;
}
private SystemUsage newSystemUsage(final Map params) {
final SystemUsage systemUsage = new SystemUsage();
{
final String memory = params.remove("systemUsage.memory.limit");
if (memory != null) {
systemUsage.getMemoryUsage().setLimit(Integer.parseInt(memory.trim()));
} else {
systemUsage.getMemoryUsage().setLimit(1024L * 1024 * 1024);
}
}
{
final String memory = params.remove("systemUsage.temp.limit");
if (memory != null) {
systemUsage.getTempUsage().setLimit(Integer.parseInt(memory.trim()));
} else {
systemUsage.getTempUsage().setLimit(1024L * 1024 * 1024 * 50);
}
}
{
final String memory = params.remove("systemUsage.store.limit");
if (memory != null) {
systemUsage.getStoreUsage().setLimit(Integer.parseInt(memory.trim()));
} else {
systemUsage.getStoreUsage().setLimit(1024L * 1024 * 1024 * 100);
}
}
{
final String memory = params.remove("systemUsage.scheduler.limit");
if (memory != null) {
systemUsage.getJobSchedulerUsage().setLimit(Integer.parseInt(memory.trim()));
} else {
systemUsage.getJobSchedulerUsage().setLimit(1024L * 1024 * 1024);
}
}
return systemUsage;
}
// forking org.apache.activemq.broker.DefaultBrokerFactory.createBroker() to support network connector properties
private BrokerService newDefaultBroker(final URI uri) throws Exception {
final URISupport.CompositeData compositeData = URISupport.parseComposite(uri);
final Map params = new HashMap<>(compositeData.getParameters());
final BrokerService brokerService = newPatchedBrokerService();
IntrospectionSupport.setProperties(brokerService, params);
if (!params.isEmpty()) {
String msg = "There are " + params.size()
+ " Broker options that couldn't be set on the BrokerService."
+ " Check the options are spelled correctly."
+ " Unknown parameters=[" + params + "]."
+ " This BrokerService cannot be started.";
throw new IllegalArgumentException(msg);
}
if (compositeData.getPath() != null) {
brokerService.setBrokerName(compositeData.getPath());
}
for (final URI component : compositeData.getComponents()) {
if ("network".equals(component.getScheme())) {
brokerService.addNetworkConnector(component.getSchemeSpecificPart());
} else if ("proxy".equals(component.getScheme())) {
brokerService.addProxyConnector(component.getSchemeSpecificPart());
} else {
brokerService.addConnector(component);
}
}
return brokerService;
}
private BrokerService newPatchedBrokerService() {
return new BrokerService() {
@Override
public NetworkConnector addNetworkConnector(final URI discoveryAddress) throws Exception {
final NetworkConnector connector = new DiscoveryNetworkConnector(discoveryAddress);
try { // try to set properties to broker too
final Map props = URISupport.parseParameters(discoveryAddress);
if (!props.containsKey("skipConnector")) {
IntrospectionSupport.setProperties(connector, props);
}
} catch (final URISyntaxException e) {
// low level cause not supported by AMQ by default
Logger.getInstance(LogCategory.OPENEJB_STARTUP, ActiveMQ5Factory.class).getChildLogger("service")
.debug(e.getMessage());
}
return addNetworkConnector(connector);
}
};
}
private BrokerPlugin[] createPlugins(final Map params) {
final String plugins = params.remove("amq.plugins");
if (plugins == null) {
return null;
}
final Collection instances = new LinkedList<>();
for (final String p : plugins.split(" *, *")) {
if (p.isEmpty()) {
continue;
}
final String prefix = p + ".";
final ObjectRecipe recipe = new ObjectRecipe(params.remove(prefix + "class"));
final Iterator> iterator = params.entrySet().iterator();
while (iterator.hasNext()) {
final Map.Entry entry = iterator.next();
final String key = entry.getKey();
if (key.startsWith(prefix)) {
recipe.setProperty(key.substring(prefix.length()), entry.getValue());
iterator.remove();
}
}
instances.add(BrokerPlugin.class.cast(recipe.create()));
}
return instances.toArray(new BrokerPlugin[instances.size()]);
}
private static String cleanUpUri(final String schemeSpecificPart, final Map parameters, final Map params) {
String uri = schemeSpecificPart;
for (final Map.Entry entry : parameters.entrySet()) {
if (!params.containsKey(entry.getKey())) {
String kv = entry.getKey() + "=" + encodeURI(entry.getValue());
int idx = uri.indexOf(kv);
if (idx < 0) {
kv = entry.getKey() + "=" + entry.getValue();
idx = uri.indexOf(kv);
}
if (idx >= 0) {
final int andIdx = idx + kv.length();
if (andIdx < uri.length() && uri.charAt(andIdx) == '&') {
uri = uri.replace(kv + "&", "");
} else {
uri = uri.replace(kv, "");
}
}
}
}
return uri;
}
private static String encodeURI(final String value) {
try {
return URLEncoder.encode(value, "UTF-8");
} catch (final UnsupportedEncodingException e) {
return value;
}
}
private static PersistenceAdapter createPersistenceAdapter(final String clazz, final String prefix, final Map params) throws IllegalAccessException, InvocationTargetException, ClassNotFoundException, InstantiationException {
Class> aClass = Thread.currentThread().getContextClassLoader().loadClass(clazz);
final PersistenceAdapter persistenceAdapter = PersistenceAdapter.class.cast(aClass.newInstance());
while (aClass != null) {
for (final Method m : aClass.getDeclaredMethods()) {
if (m.getName().startsWith("set") && m.getParameterTypes().length == 1 && Modifier.isPublic(m.getModifiers())) {
final String key = prefix + "." + m.getName().substring(3).toLowerCase(Locale.ENGLISH);
final Object field = params.remove(key);
if (field != null) {
try {
final Object toSet = propertyEditorRegistry.getValue(m.getParameterTypes()[0], field.toString());
m.invoke(persistenceAdapter, toSet);
} catch (final PropertyEditorException cantConvertException) {
throw new IllegalArgumentException("can't convert " + field + " for " + m.getName(), cantConvertException);
}
}
}
}
aClass = aClass.getSuperclass();
}
return persistenceAdapter;
}
private void tomeeConfig(final BrokerService broker) {
//Notify when an error occurs on shutdown.
broker.setUseLoggingForShutdownErrors(Logger.getInstance(LogCategory.OPENEJB_STARTUP, ActiveMQ5Factory.class).isErrorEnabled());
}
private Properties getLowerCaseProperties() {
final Properties newProperties = new Properties();
if (properties != null) {
Object key;
for (final Map.Entry
© 2015 - 2025 Weber Informatics LLC | Privacy Policy