Package org.exoplatform.services.jcr.infinispan

Source Code of org.exoplatform.services.jcr.infinispan.ISPNCacheFactory

/*
* Copyright (C) 2010 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.jcr.infinispan;

import org.apache.commons.io.IOUtils;
import org.exoplatform.commons.utils.SecurityHelper;
import org.exoplatform.container.ExoContainer;
import org.exoplatform.container.ExoContainerContext;
import org.exoplatform.container.configuration.ConfigurationManager;
import org.exoplatform.services.jcr.config.MappedParametrizedObjectEntry;
import org.exoplatform.services.jcr.config.RepositoryConfigurationException;
import org.exoplatform.services.jcr.config.TemplateConfigurationHelper;
import org.exoplatform.services.log.ExoLogger;
import org.exoplatform.services.log.Log;
import org.infinispan.Cache;
import org.infinispan.config.Configuration;
import org.infinispan.config.GlobalConfiguration;
import org.infinispan.jmx.MBeanServerLookup;
import org.infinispan.manager.DefaultCacheManager;
import org.infinispan.manager.EmbeddedCacheManager;

import java.io.IOException;
import java.io.InputStream;
import java.net.URL;
import java.security.PrivilegedAction;
import java.security.PrivilegedExceptionAction;
import java.util.HashMap;
import java.util.Map;
import java.util.Properties;

import javax.management.MBeanServer;

/**
* Factory that creates and starts pre-configured instances of Infinispan.
* Path to Infinispan configuration or template and cache name should be
* provided as "infinispan-configuration" and "infinispan-cache-name" properties
* in parameterEntry instance respectively.
*
* @author <a href="mailto:nikolazius@gmail.com">Nikolay Zamosenchuk</a>
* @version $Id: InfinispanCacheFactory.java 3001 2010-08-30 06:56:05Z tolusha $
*
*/
public class ISPNCacheFactory<K, V>
{

   public static final String INFINISPAN_CONFIG = "infinispan-configuration";

   private final ConfigurationManager configurationManager;
   private final TemplateConfigurationHelper configurationHelper;

   private static final Log log = ExoLogger.getLogger("exo.jcr.component.core.InfinispanCacheFactory");

   /**
    * A Map that contains all the registered CacheManager order by {@link ExoContainer}
    * instances and {@link GlobalConfiguration}.
    */
   private static Map<GlobalConfiguration, EmbeddedCacheManager> CACHE_MANAGERS =
      new HashMap<GlobalConfiguration, EmbeddedCacheManager>();

   private static final MBeanServerLookup MBEAN_SERVER_LOOKUP = new MBeanServerLookup()
   {
      public MBeanServer getMBeanServer(Properties properties)
      {
         return ExoContainerContext.getTopContainer().getMBeanServer();
      }     
   };
  
   /**
    * Creates InfinispanCacheFactory with provided configuration transaction managers.
    * Transaction manager will later be injected to cache instance.
    *
    * @param configurationManager
    */
   public ISPNCacheFactory(ConfigurationManager configurationManager)
   {
      this.configurationManager = configurationManager;
      this.configurationHelper = new ISPNCacheHelper(configurationManager);
   }

   /**
   * Factory that creates and starts pre-configured instances of Infinispan.
   * Path to Infinispan configuration or template should be provided as
   * "infinispan-configuration" property in parameterEntry instance.
   * <br>
   *
   * @param regionId the unique id of the cache region to create
   * @param parameterEntry
   * @return
   * @throws RepositoryConfigurationException
   */
   public Cache<K, V> createCache(final String regionId, MappedParametrizedObjectEntry parameterEntry)
      throws RepositoryConfigurationException
   {
      // get Infinispan configuration file path
      final String configurationPath = parameterEntry.getParameterValue(INFINISPAN_CONFIG);
      log.info("Infinispan Cache configuration used: " + configurationPath);
      // avoid dashes in cache name. Some SQL servers doesn't allow dashes in table names
      final String regionIdEscaped = regionId.replace("-", "_");
      // prepare configuration
      final InputStream configStream;
      try
      {
         // fill template
         configStream = configurationHelper.fillTemplate(configurationPath, parameterEntry.getParameters());
      }
      catch (IOException e)
      {
         throw new RepositoryConfigurationException(e);
      }

      // create cache
      final EmbeddedCacheManager manager;
      try
      {
         // creating new CacheManager using SecurityHelper

         manager = SecurityHelper.doPrivilegedIOExceptionAction(new PrivilegedExceptionAction<EmbeddedCacheManager>()
         {
            public EmbeddedCacheManager run() throws IOException
            {
               DefaultCacheManager manager = new DefaultCacheManager(configStream, false);
               loadJGroupsConfig(manager);
               return getUniqueInstance(regionIdEscaped, manager);
            }
         });

      }
      catch (IOException e)
      {
         throw new RepositoryConfigurationException(e);
      }

      PrivilegedAction<Cache<K, V>> action = new PrivilegedAction<Cache<K, V>>()
      {
         public Cache<K, V> run()
         {
            return manager.getCache(regionIdEscaped);
         }
      };
      Cache<K, V> cache = SecurityHelper.doPrivilegedAction(action);

      return cache;
   }

   /**
    * This method is used to load the file corresponding to the path set on the property
    * <tt>configurationFile</tt> using the configuration manager, then set the XML content
    *  as value of the property <tt>configurationXml</tt>.
    * @param manager the manager from which we extract the transport properties in which
    * we will find the path of the JGroups configuration
    * @throws IOException if the configuration file cannot be read
    */
   private void loadJGroupsConfig(DefaultCacheManager manager) throws IOException
   {
      Properties p = manager.getGlobalConfiguration().getTransportProperties();
      if (p != null && p.containsKey("configurationFile"))
      {
         URL jgroupsConfigURL = null;
         InputStream jgroupsConfigInputStream = null;
         try
         {
            // Trying to get the configuration from the configuration manager
            String configurationFile = p.getProperty("configurationFile");
            jgroupsConfigInputStream = configurationManager.getInputStream(configurationFile);
            jgroupsConfigURL = configurationManager.getResource(configurationFile);
         }
         catch (Exception e)
         {
            // ignore me
         }
         if (jgroupsConfigInputStream != null)
         {
            try
            {
               log.info("Custom JGroups configuration set: " + jgroupsConfigURL);
               p.setProperty("configurationXml", IOUtils.toString(jgroupsConfigInputStream));
               p.remove("configurationFile");
            }
            finally
            {
               jgroupsConfigInputStream.close();
            }
         }
      }
   }

   /**
    * Try to find if a {@link EmbeddedCacheManager} of the same type (i.e. their {@link GlobalConfiguration} are equals)
    * has already been registered for the same current container.
    * If no cache manager has been registered, we register the given cache manager otherwise we
    * use the previously registered cache manager and we define a dedicated region for the related cache.
    * @param regionId the unique id of the cache region to create
    * @param manager the current cache manager of the cache to create
    * @return the given cache manager if it has not been registered otherwise the cache manager of the same
    * type that has already been registered..
    */
   private static synchronized EmbeddedCacheManager getUniqueInstance(String regionId, EmbeddedCacheManager manager)
   {
      GlobalConfiguration gc = manager.getGlobalConfiguration();
      ExoContainer container = ExoContainerContext.getCurrentContainer();
      // Ensure that the cluster name won't be used between 2 ExoContainers
      gc.setClusterName(gc.getClusterName() + "_" + container.getContext().getName());
      gc.setCacheManagerName(gc.getCacheManagerName() + "_" + container.getContext().getName());
      // Configure the MBeanServerLookup
      gc.setMBeanServerLookupInstance(MBEAN_SERVER_LOOKUP);
      Configuration conf = manager.getDefaultConfiguration();
      if (CACHE_MANAGERS.containsKey(gc))
      {
         manager = CACHE_MANAGERS.get(gc);
      }
      else
      {
         // Reset the manager before storing it into the map since the default config is used as
         // template to define a new configuration
         manager = new DefaultCacheManager(gc);
         CACHE_MANAGERS.put(gc, manager);
         if (log.isInfoEnabled())
         {
            log.info("A new ISPN Cache Manager instance has been registered for the region " + regionId
               + " and the container " + container.getContext().getName());
         }
      }
      // Define the configuration of the cache
      manager.defineConfiguration(regionId, conf);
      if (log.isInfoEnabled())
      {
         log.info("The region " + regionId + " has been registered for the container "
            + container.getContext().getName());
      }
      return manager;
   }
}
TOP

Related Classes of org.exoplatform.services.jcr.infinispan.ISPNCacheFactory

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.