/** * 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.portal.pom.config.cache; import java.io.Serializable; import java.util.concurrent.atomic.AtomicLong; import org.exoplatform.container.ExoContainer; import org.exoplatform.portal.pom.config.POMSession; import org.exoplatform.portal.pom.config.POMTask; import org.exoplatform.portal.pom.config.TaskExecutionDecorator; import org.exoplatform.portal.pom.config.TaskExecutor; import org.gatein.common.logging.Logger; import org.gatein.common.logging.LoggerFactory; /** * @author <a href="mailto:julien.viet@exoplatform.com">Julien Viet</a> * @version $Revision$ */ public class DataCache extends TaskExecutionDecorator { /** . */ private final Logger log = LoggerFactory.getLogger(DataCache.class); /** . */ private final AtomicLong readCount = new AtomicLong(); /** . */ private boolean cluster = ExoContainer.getProfiles().contains("cluster"); public DataCache(TaskExecutor next) { super(next); } public <V> V execute(POMSession session, POMTask<V> task) throws Exception { if (task instanceof CacheableDataTask) { CacheableDataTask<?, V> loadTask = (CacheableDataTask<?, V>) task; switch (loadTask.getAccessMode()) { case READ: return read(session, loadTask); case CREATE: return create(session, loadTask); case WRITE: return write(session, loadTask); case DESTROY: return remove(session, loadTask); default: throw new UnsupportedOperationException(); } } else { return super.execute(session, task); } } private <K extends Serializable, V> V remove(POMSession session, CacheableDataTask<K, V> task) throws Exception { K key = task.getKey(); if (log.isTraceEnabled()) { log.trace("Schedule " + key + " for eviction"); } session.scheduleForEviction(key); return super.execute(session, task); } private <K extends Serializable, V> V write(POMSession session, CacheableDataTask<K, V> task) throws Exception { K key = task.getKey(); if (log.isTraceEnabled()) { log.trace("Schedule " + key + " for eviction"); } session.scheduleForEviction(key); return super.execute(session, task); } private <K extends Serializable, V> V create(POMSession session, CacheableDataTask<K, V> task) throws Exception { K key = task.getKey(); if (log.isTraceEnabled()) { log.trace("Schedule " + key + " for eviction"); } session.scheduleForEviction(key); return super.execute(session, task); } private <K extends Serializable, V> V read(POMSession session, CacheableDataTask<K, V> task) throws Exception { K key = task.getKey(); // if (!session.isModified()) { Object o = session.getFromCache(key); if (log.isTraceEnabled()) { log.trace("Retrieved " + o + " for key " + key); } V v = null; if (o != null) { if (o == NullObject.get()) { if (log.isTraceEnabled()) { log.trace("Returning null as found null object marker"); } return null; } else { Class<V> type = task.getValueType(); if (type.isInstance(o)) { v = type.cast(o); } else { log.error("Object " + o + " was not of the expected type " + type); } } } // if (v != null) { if (log.isTraceEnabled()) { log.trace("Returning object " + v + " for key " + key); } return v; } else { readCount.incrementAndGet(); // if (log.isTraceEnabled()) { log.trace("Object not found in cache for key " + key + " about to retrieve it"); } // v = super.execute(session, task); if (log.isTraceEnabled()) { log.trace("Retrieved object " + v + " key " + key + " that will be returned"); } // if (!session.isModified()) { if (v == null) { if (log.isTraceEnabled()) { log.trace("Updating cache with null object for key " + key); } session.putInCache(key, NullObject.get()); } else { if (log.isTraceEnabled()) { log.trace("Updating cache with object " + v + " for key " + key); } session.putInCache(key, v); } } // return v; } } else { if (log.isTraceEnabled()) { log.trace("Session was modified, object for key " + key + " is directly retrieved"); } // return super.execute(session, task); } } public long getReadCount() { return readCount.longValue(); } }