/*
* Copyright (C) 2009 eXo Platform SAS.
*
* This is free software; you can redistribute it and/or modify it
* under the terms of the GNU Lesser General Public License as
* published by the Free Software Foundation; either version 2.1 of
* the License, or (at your option) any later version.
*
* This software is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this software; if not, write to the Free
* Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
* 02110-1301 USA, or see the FSF site: http://www.fsf.org.
*/
package org.exoplatform.services.cache.impl.jboss;
import org.exoplatform.container.ExoContainer;
import org.exoplatform.container.ExoContainerContext;
import org.exoplatform.container.configuration.ConfigurationManager;
import org.exoplatform.container.xml.InitParams;
import org.exoplatform.container.xml.ValueParam;
import org.exoplatform.services.cache.ExoCache;
import org.exoplatform.services.cache.ExoCacheConfig;
import org.exoplatform.services.cache.ExoCacheFactory;
import org.exoplatform.services.cache.ExoCacheInitException;
import org.exoplatform.services.cache.impl.jboss.fifo.FIFOExoCacheCreator;
import org.exoplatform.services.log.ExoLogger;
import org.exoplatform.services.log.Log;
import org.jboss.cache.Cache;
import org.jboss.cache.CacheFactory;
import org.jboss.cache.DefaultCacheFactory;
import org.jboss.cache.config.Configuration;
import org.jboss.cache.config.Configuration.CacheMode;
import org.jboss.cache.config.EvictionConfig;
import org.jboss.cache.config.EvictionRegionConfig;
import org.jboss.cache.jmx.JmxRegistrationManager;
import org.picocontainer.Startable;
import java.io.Serializable;
import java.util.HashMap;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;
import java.util.concurrent.CopyOnWriteArrayList;
import javax.management.ObjectName;
/**
* This class is the JBoss Cache implementation of the {@link org.exoplatform.services.cache.ExoCacheFactory}
* Created by The eXo Platform SAS
* Author : eXoPlatform
* exo@exoplatform.com
* 17 juil. 2009
*/
public class ExoCacheFactoryImpl implements ExoCacheFactory, Startable
{
/**
* The logger
*/
private static final Log LOG = ExoLogger.getLogger("exo.kernel.component.ext.cache.impl.jboss.v3.ExoCacheFactoryImpl");
/**
* The initial parameter key that defines the full path of the configuration template
*/
private static final String CACHE_CONFIG_TEMPLATE_KEY = "cache.config.template";
/**
* The configuration manager that allows us to retrieve a configuration file in several different
* manners
*/
private final ConfigurationManager configManager;
/**
* The full path of the configuration template
*/
private final String cacheConfigTemplate;
/**
* The mapping between the configuration types and the creators
*/
private final Map<Class<? extends ExoCacheConfig>, ExoCacheCreator> mappingConfigTypeCreators =
new HashMap<Class<? extends ExoCacheConfig>, ExoCacheCreator>();
/**
* The mapping between the implementations and the creators. This is mainly used for backward compatibility
*/
private final Map<String, ExoCacheCreator> mappingImplCreators = new HashMap<String, ExoCacheCreator>();
/**
* The mapping between the cache names and the configuration paths
*/
private final Map<String, String> mappingCacheNameConfig = new HashMap<String, String>();
/**
* The list of all the JMX Managers registered
*/
private final List<JmxRegistrationManager> jmxManagers = new CopyOnWriteArrayList<JmxRegistrationManager>();
/**
* The default creator
*/
private final ExoCacheCreator defaultCreator = new FIFOExoCacheCreator();
public ExoCacheFactoryImpl(InitParams params, ConfigurationManager configManager)
{
this.configManager = configManager;
this.cacheConfigTemplate = getValueParam(params, CACHE_CONFIG_TEMPLATE_KEY);
if (cacheConfigTemplate == null)
{
throw new RuntimeException("The parameter '" + CACHE_CONFIG_TEMPLATE_KEY + "' must be set");
}
}
/**
* To create a new cache instance according to the given configuration, we follow the steps below:
*
* 1. We first try to find if a specific location of the cache configuration has been defined thanks
* to an external component plugin of type ExoCacheFactoryConfigPlugin
* 2. If no specific location has been defined, we use the default configuration which is
* "${CACHE_CONFIG_TEMPLATE_KEY}"
*/
public ExoCache<Serializable, Object> createCache(ExoCacheConfig config) throws ExoCacheInitException
{
final String region = config.getName();
final String customConfig = mappingCacheNameConfig.get(region);
final Cache<Serializable, Object> cache;
final CacheFactory<Serializable, Object> factory = new DefaultCacheFactory<Serializable, Object>();
final ExoCache<Serializable, Object> eXoCache;
try
{
if (customConfig != null)
{
// A custom configuration has been set
if (LOG.isInfoEnabled())
LOG.info("A custom configuration has been set for the cache '" + region + "'.");
cache = factory.createCache(configManager.getInputStream(customConfig), false);
}
else
{
// No custom configuration has been found, a configuration template will be used
if (LOG.isInfoEnabled())
LOG.info("The configuration template will be used for the the cache '" + region + "'.");
cache = factory.createCache(configManager.getInputStream(cacheConfigTemplate), false);
if (!config.isDistributed())
{
// The cache is local
cache.getConfiguration().setCacheMode(CacheMode.LOCAL);
}
// Re initialize the template to avoid conflicts
cleanConfigurationTemplate(cache, region);
}
final ExoCacheCreator creator = getExoCacheCreator(config);
// Create the cache
eXoCache = creator.create(config, cache);
// Create the cache
cache.create();
// Start the cache
cache.start();
// Create JMX Manager for this cache instance
JmxRegistrationManager jmxManager = getJmxRegistrationManager(cache, region);
if (jmxManager != null)
{
jmxManager.registerAllMBeans();
jmxManagers.add(jmxManager);
}
}
catch (Exception e)
{
throw new ExoCacheInitException("The cache '" + region + "' could not be initialized", e);
}
return eXoCache;
}
/**
* Gives the {@link JmxRegistrationManager} instance corresponding to the given context
*/
private static JmxRegistrationManager getJmxRegistrationManager(Cache<?, ?> parentCache,
String cacheName)
{
try
{
ExoContainer container = ExoContainerContext.getCurrentContainer();
ObjectName containerObjectName = container.getScopingObjectName();
final String objectNameBase = (containerObjectName != null ? containerObjectName.toString() + "," : "exo:")+ "cache-name=" + cacheName;
return new JmxRegistrationManager(container.getMBeanServer(), parentCache, objectNameBase)
{
public String getObjectName(String resourceName)
{
return objectNameBase + JMX_RESOURCE_KEY + resourceName;
}
};
}
catch (Exception e)
{
LOG.error("Could not create the JMX Manager", e);
}
return null;
}
/**
* Add a list of creators to register
* @param plugin the plugin that contains the creators
*/
public void addCreator(ExoCacheCreatorPlugin plugin)
{
final List<ExoCacheCreator> creators = plugin.getCreators();
for (ExoCacheCreator creator : creators)
{
mappingConfigTypeCreators.put(creator.getExpectedConfigType(), creator);
mappingImplCreators.put(creator.getExpectedImplementation(), creator);
}
}
/**
* Add a list of custom configuration to register
* @param plugin the plugin that contains the configs
*/
public void addConfig(ExoCacheFactoryConfigPlugin plugin)
{
final Map<String, String> configs = plugin.getConfigs();
mappingCacheNameConfig.putAll(configs);
}
/**
* Returns the value of the ValueParam if and only if the value is not empty
*/
private static String getValueParam(InitParams params, String key)
{
if (params == null)
{
return null;
}
final ValueParam vp = params.getValueParam(key);
String result;
if (vp == null || (result = vp.getValue()) == null || (result = result.trim()).length() == 0)
{
return null;
}
return result;
}
/**
* Returns the most relevant ExoCacheCreator according to the give configuration
*/
protected ExoCacheCreator getExoCacheCreator(ExoCacheConfig config)
{
ExoCacheCreator creator = mappingConfigTypeCreators.get(config.getClass());
if (creator == null)
{
// No creator for this type has been found, let's try the implementation field
creator = mappingImplCreators.get(config.getImplementation());
if (creator == null)
{
// No creator can be found, we will use the default creator
if (LOG.isInfoEnabled())
LOG.info("No cache creator has been found for the the cache '" + config.getName()
+ "', the default one will be used.");
return defaultCreator;
}
}
if (LOG.isInfoEnabled())
LOG.info("The cache '" + config.getName() + "' will be created with '" + creator.getClass() + "'.");
return creator;
}
/**
* Clean the configuration template to prevent conflicts
*/
protected void cleanConfigurationTemplate(Cache<Serializable, Object> cache, String region)
{
final Configuration config = cache.getConfiguration();
// Reset the eviction policies
EvictionConfig evictionConfig = config.getEvictionConfig();
if (evictionConfig == null)
{
// If not eviction config exists, we create an empty one
evictionConfig = new EvictionConfig();
config.setEvictionConfig(evictionConfig);
}
evictionConfig.setEvictionRegionConfigs(new LinkedList<EvictionRegionConfig>());
// Rename the cluster name
String clusterName = config.getClusterName();
if (clusterName != null && (clusterName = clusterName.trim()).length() > 0)
{
config.setClusterName(clusterName + " " + region);
}
}
/**
* @see org.picocontainer.Startable#start()
*/
public void start()
{
}
/**
* @see org.picocontainer.Startable#stop()
*/
public void stop()
{
for (JmxRegistrationManager jmxManager : jmxManagers)
{
jmxManager.unregisterAllMBeans();
}
}
}