PortletFutureCache.java

/*
 * 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.portletcache;

import java.util.Iterator;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.ScheduledFuture;
import java.util.concurrent.TimeUnit;

import org.exoplatform.commons.cache.future.FutureCache;
import org.exoplatform.services.log.Log;

/**
 * @author <a href="mailto:julien.viet@exoplatform.com">Julien Viet</a>
 * @version $Revision$
 */
@Deprecated
class PortletFutureCache extends FutureCache<WindowKey, MarkupFragment, PortletRenderContext>
{

  /**
   * default to 5000 entries
   */
  private static final int                                   DEFAULT_CACHE_SIZE = 5000;

  /** . */
  private final ConcurrentHashMap<WindowKey, MarkupFragment> entries;

  /** . */
  private final Log                                          log;

  /** . */
  private ScheduledExecutorService                           scheduler          = Executors.newScheduledThreadPool(1);

  /** . */
  private int                                                cleanupCache;

  /** . */
  private int                                                cacheMaxSize       = DEFAULT_CACHE_SIZE;

  /** . */
  private ScheduledFuture<?>                                 scheduled;

  PortletFutureCache(Log log, int cleanupCache)
  {
    super(new PortletRenderer(log));

    //
    this.log = log;
    this.entries = new ConcurrentHashMap<WindowKey, MarkupFragment>();
    this.cleanupCache = preventWrongCleanupCacheValue(cleanupCache);
    this.scheduled = null;
  }

  PortletFutureCache(Log log, int cleanupCache, int cacheSize)
  {
    this(log, cleanupCache);
    this.cacheMaxSize = cacheSize;
  }

  private static int preventWrongCleanupCacheValue(int value)
  {
    // 10 mns by default
    return value < 0 ? 5 * 60 : value;
  }

  public int getCleanupCache()
  {
    return cleanupCache;
  }

  public void updateCleanupCache(int cleanupCache)
  {
    this.cleanupCache = cleanupCache;

    //
    if (scheduled != null)
    {
      stop();
      start();
    }
  }

  /*
   * Returns the number of entries in this cache.
   */
  protected int getCacheSize()
  {
    return (entries.size());
  }

  /*
   * Returns the defined Max Cache Size.
   */
  protected int getCacheMaxSize()
  {
    return (cacheMaxSize);
  }

  /*
   * Returns the defined Max Cache Size.
   */
  protected void setCacheMaxSize(int cacheMaxSize)
  {
    this.cacheMaxSize = cacheMaxSize;
  }

  /*
   * Clear Cache (should be called by JMX in urgently mode.
   */
  protected void clearCache()
  {
    if (scheduled != null)
    {
      stop();
      entries.clear();
      start();
    }
    else
      entries.clear();
  }

  /*
   * Did Cache contains the following key (do not remove old values).
   */
  protected boolean containsKey(WindowKey key)
  {
    return (entries.containsKey(key));
  }


  @Override
  protected MarkupFragment get(WindowKey key)
  {
    // System.out.println("get asked for key " + key);

    MarkupFragment value = entries.get(key);
    if (value != null)
    {
      if (value.expirationTimeMillis > System.currentTimeMillis())
      {
        if (log.isTraceEnabled())
          log.trace("Using cached markup for portlet " + key);
        return value;
      }
      if (log.isTraceEnabled())
        log.trace("Expired markup for portlet " + key);
      entries.remove(key);
      return null;
    }
    return null;
  }

  @Override
  protected void put(WindowKey key, MarkupFragment value)
  {
    boolean canInsert = false;

    // System.out.println("put asked for key " + key + " duration " + (value.expirationTimeMillis - System.currentTimeMillis()));

    if (value.expirationTimeMillis > System.currentTimeMillis()) {
      if ((entries.size() < cacheMaxSize))
        canInsert = true;

      if (canInsert) {
        entries.put(key, value);
        if (log.isTraceEnabled())
          log.trace("Cached markup for portlet " + key);

        // System.out.println("Cached markup for portlet " + key);
      }
    }
  }

  @Override
  protected void putOnly(WindowKey key, MarkupFragment value) {
    put(key, value);
  }

  public void start()
  {
    if (scheduled == null)
    {
      if (log.isDebugEnabled()) {
        log.debug("Starting cache cleaner with a period of " + cleanupCache + " seconds");
      }
      if(scheduler.isShutdown()) scheduler = Executors.newScheduledThreadPool(1);
      scheduled = scheduler.scheduleWithFixedDelay(new Runnable()
      {
        public void run()
        {
          long now = System.currentTimeMillis();
          for (Iterator<Map.Entry<WindowKey, MarkupFragment>> i = entries.entrySet().iterator(); i.hasNext();)
          {
            Map.Entry<WindowKey, MarkupFragment> entry = i.next();
            if (entry.getValue().expirationTimeMillis < now)
            {
              if (log.isTraceEnabled())
                log.trace("Removing expired entry " + entry.getKey().getWindowId());

              // System.out.println("Removing expired entry " + entry.getKey().getWindowId());

              i.remove();
            }
          }
        }
      }, cleanupCache, cleanupCache, TimeUnit.SECONDS);
    }
  }

  public void stop()
  {
    if (scheduled != null)
    {
      if (log.isDebugEnabled()) {
        log.debug("Stopping cache cleaner");
      }
      scheduled.cancel(false);
      scheduled = null;
    }
    if (scheduler != null) {
      scheduler.shutdown();
    }
  }
}