/* * JBoss, Home of Professional Open Source. * Copyright 2008, Red Hat Middleware LLC, and individual contributors * as indicated by the @author tags. See the copyright.txt file in the * distribution for a full listing of individual contributors. * * 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.jboss.ha.framework.server.util; import java.io.Serializable; import java.util.Collection; import java.util.Iterator; import java.util.Timer; import java.util.TimerTask; import org.jboss.ha.framework.interfaces.DistributedState; import org.jboss.ha.framework.interfaces.HAPartition; import org.jboss.ha.framework.server.HAPartitionLocator; import org.jboss.logging.Logger; import org.jboss.util.CachePolicy; /** An implementation of a timed cache. This is a cache whose entries have a limited lifetime with the ability to refresh their lifetime. The entries managed by the cache implement the TimedCachePolicy.TimedEntry interface. If an object inserted into the cache does not implement this interface, it will be wrapped in a DefaultTimedEntry and will expire without the possibility of refresh after getDefaultLifetime() seconds. This is a lazy cache policy in that objects are not checked for expiration until they are accessed. @author <a href="mailto:Scott.Stark@jboss.org">Scott Stark</a>. @version $Revision: 81001 $ */ public class DistributedTimedCachePolicy extends TimerTask implements CachePolicy { /** The interface that cache entries support. */ public static interface TimedEntry extends Serializable { /** Initializes an entry with the current cache time. This is called when the entry is first inserted into the cache so that entries do not have to know the absolute system time. */ public void init(long now); /** Is the entry still valid basis the current time @return true if the entry is within its lifetime, false if it is expired. */ public boolean isCurrent(long now); /** Attempt to extend the entry lifetime by refreshing it. @return true if the entry was refreshed successfully, false otherwise. */ public boolean refresh(); /** Notify the entry that it has been removed from the cache. */ public void destroy(); /** Get the value component of the TimedEntry. This may or may not be the TimedEntry implementation. */ public Object getValue(); } protected static Timer resolutionTimer = new Timer(true); protected static Logger log = Logger.getLogger(DistributedTimedCachePolicy.class); /** The map of cached TimedEntry objects. */ protected DistributedState entryMap; protected String category; protected String partitionName; /** The lifetime in seconds to use for objects inserted that do not implement the TimedEntry interface. */ protected volatile int defaultLifetime; /** The caches notion of the current time */ protected volatile long now; /** The resolution in seconds of the cache current time */ protected int resolution; /** Creates a new TimedCachePolicy with the given default entry lifetime that does not synchronized access to its policy store and uses a 60 second resolution. */ public DistributedTimedCachePolicy(String category, String partitionName, int defaultLifetime) { this(category, partitionName, defaultLifetime, 0); } /** Creates a new TimedCachePolicy with the given default entry lifetime that does/does not synchronized access to its policy store depending on the value of threadSafe. @param category the name of the catetegory used in the DistributedState access calls. @param partitionName the name of the HAPartition who's replicated state service will be used as the cache store. @param defaultLifetime the lifetime in seconds to use for objects inserted that do not implement the TimedEntry interface. @param resolution the resolution in seconds of the cache timer. A cache does not query the system time on every get() invocation. Rather the cache updates its notion of the current time every 'resolution' seconds. @see DistributedState */ public DistributedTimedCachePolicy(String category, String partitionName, int defaultLifetime, int resolution) { this.category = category; this.partitionName = partitionName; this.defaultLifetime = defaultLifetime; this.resolution = (resolution > 0) ? resolution : 60; } // Service implementation ---------------------------------------------- /** Initializes the cache for use. Prior to this the cache has no store. */ public void create() { // Lookup the parition HAPartition partition = HAPartitionLocator.getHAPartitionLocator().getHAPartition(this.partitionName, null); this.entryMap = partition.getDistributedStateService(); log.debug("Obtained DistributedState from partition=" + this.partitionName); this.now = System.currentTimeMillis(); } /** Schedules this with the class resolutionTimer Timer object for execution every resolution seconds. */ public void start() { resolutionTimer.scheduleAtFixedRate(this, 0, 1000 * this.resolution); } /** Stop cancels the resolution timer and flush()es the cache. */ public void stop() { super.cancel(); } /** Clears the cache of all entries. */ public void destroy() { } // --- Begin CachePolicy interface methods /** Get the cache value for key if it has not expired. If the TimedEntry is expired its destroy method is called and then removed from the cache. @return the TimedEntry value or the original value if it was not an instance of TimedEntry if key is in the cache, null otherwise. */ public Object get(Object key) { Serializable skey = (Serializable) key; TimedEntry entry = (TimedEntry) this.entryMap.get(this.category, skey); if (entry == null) return null; if (entry.isCurrent(this.now) == false) { // Try to refresh the entry if (entry.refresh() == false) { // Failed, remove the entry and return null entry.destroy(); try { this.entryMap.remove(this.category, skey); } catch (Exception e) { log.debug("Failed to remove expired entry", e); } return null; } } Object value = entry.getValue(); return value; } /** Get the cache value for key. This method does not check to see if the entry has expired. @return the TimedEntry value or the original value if it was not an instancee of TimedEntry if key is in the cache, null otherwise. */ public Object peek(Object key) { Serializable skey = (Serializable) key; TimedEntry entry = (TimedEntry) this.entryMap.get(this.category, skey); Object value = null; if (entry != null) { value = entry.getValue(); } return value; } /** Insert a value into the cache. In order to have the cache entry reshresh itself value would have to implement TimedEntry and implement the required refresh() method logic. @param key the key for the cache entry @param value Either an instance of TimedEntry that will be inserted without change, or an abitrary value that will be wrapped in a non-refreshing TimedEntry. */ public void insert(Object key, Object value) { Serializable skey = (Serializable) key; TimedEntry entry = (TimedEntry) this.entryMap.get(this.category, skey); if (entry != null) { throw new IllegalStateException("Attempt to insert duplicate entry"); } if ((value instanceof TimedEntry) == false) { // Wrap the value in a DefaultTimedEntry Serializable svalue = (Serializable) value; entry = new DefaultTimedEntry(this.defaultLifetime, svalue); } else { entry = (TimedEntry) value; } entry.init(this.now); try { this.entryMap.set(this.category, skey, entry); } catch (Exception e) { log.error("Failed to set entry", e); } } /** Remove the entry associated with key and call destroy on the entry if found. */ public void remove(Object key) { Serializable skey = (Serializable) key; try { TimedEntry entry = (TimedEntry) this.entryMap.remove(this.category, skey); if (entry != null) { entry.destroy(); } } catch (Exception e) { log.error("Failed to remove entry", e); } } /** Remove all entries from the cache. */ public void flush() { Collection keys = this.entryMap.getAllKeys(this.category); // Notify the entries of their removal Iterator iter = keys.iterator(); while (iter.hasNext()) { Serializable key = (Serializable) iter.next(); TimedEntry entry = (TimedEntry) this.entryMap.get(this.category, key); entry.destroy(); } } public int size() { return this.entryMap.getAllKeys(this.category).size(); } // --- End CachePolicy interface methods /** Get the default lifetime of cache entries. @return default lifetime in seconds of cache entries. */ public int getDefaultLifetime() { return this.defaultLifetime; } /** Set the default lifetime of cache entries for new values added to the cache. @param defaultLifetime lifetime in seconds of cache values that do not implement TimedEntry. */ public void setDefaultLifetime(int defaultLifetime) { this.defaultLifetime = defaultLifetime; } /** The TimerTask run method. It updates the cache time to the current system time. */ @Override public void run() { this.now = System.currentTimeMillis(); } /** Get the cache time. @return the cache time last obtained from System.currentTimeMillis() */ public long currentTimeMillis() { return this.now; } /** Get the raw TimedEntry for key without performing any expiration check. @return the TimedEntry value associated with key if one exists, null otherwise. */ public TimedEntry peekEntry(Object key) { Serializable skey = (Serializable) key; TimedEntry entry = (TimedEntry) this.entryMap.get(this.category, skey); return entry; } /** The default implementation of TimedEntry used to wrap non-TimedEntry objects inserted into the cache. */ static class DefaultTimedEntry implements TimedEntry { long expirationTime; Serializable value; DefaultTimedEntry(long lifetime, Serializable value) { this.expirationTime = 1000 * lifetime; this.value = value; } public void init(long now) { this.expirationTime += now; } public boolean isCurrent(long now) { return this.expirationTime > now; } public boolean refresh() { return false; } public void destroy() { } public Object getValue() { return this.value; } } }