Package org.infinispan.interceptors

Source Code of org.infinispan.interceptors.ActivationInterceptor

package org.infinispan.interceptors;

import org.infinispan.commands.FlagAffectedCommand;
import org.infinispan.commands.write.ApplyDeltaCommand;
import org.infinispan.commands.write.PutMapCommand;
import org.infinispan.configuration.cache.Configuration;
import org.infinispan.configuration.cache.EvictionConfiguration;
import org.infinispan.container.DataContainer;
import org.infinispan.container.entries.CacheEntry;
import org.infinispan.container.entries.InternalCacheEntry;
import org.infinispan.context.InvocationContext;
import org.infinispan.eviction.ActivationManager;
import org.infinispan.eviction.EvictionStrategy;
import org.infinispan.factories.annotations.Inject;
import org.infinispan.factories.annotations.Start;
import org.infinispan.interceptors.locking.ClusteringDependentLogic;
import org.infinispan.util.logging.Log;
import org.infinispan.util.logging.LogFactory;

import java.util.Arrays;

public class ActivationInterceptor extends CacheLoaderInterceptor {

   private static final Log log = LogFactory.getLog(ActivationInterceptor.class);

   private Configuration cfg;
   private boolean isManualEviction;
   private ActivationManager activationManager;
   private ClusteringDependentLogic cdl;
   private DataContainer dataContainer;

   @Override
   protected Log getLog() {
      return log;
   }

   @Inject
   public void inject(Configuration cfg, ActivationManager activationManager, ClusteringDependentLogic cdl,
                      DataContainer dataContainer) {
      this.cfg = cfg;
      this.activationManager = activationManager;
      this.cdl = cdl;
      this.dataContainer = dataContainer;
   }

   @Start(priority = 15)
   @SuppressWarnings("unused")
   public void start() {
      // Treat caches configured with manual eviction differently.
      // These caches require activation at the interceptor level.
      EvictionConfiguration evictCfg = cfg.eviction();
      isManualEviction = evictCfg.strategy() == EvictionStrategy.NONE
            || evictCfg.maxEntries() < 0;
   }

   @Override
   public Object visitPutMapCommand(InvocationContext ctx, PutMapCommand command) throws Throwable {
      // Load the keys for our map into the data container that we are using removing them from the store.
      // This way when we overwrite the values on commit they won't be in loader and if we rollback they won't be
      // in the loader either but will be in data container at least
      for (Object key : command.getAffectedKeys()) {
         loadIfNeeded(ctx, key, false, command);
      }
      return super.visitPutMapCommand(ctx, command);
   }

   @Override
   protected Boolean loadIfNeeded(InvocationContext ctx, Object key, boolean isRetrieval, FlagAffectedCommand cmd) throws Throwable {
      CacheEntry entry = ctx.lookupEntry(key);
      if (entry != null && !entry.isNull() && entry.getValue() != null) {
         //the entry is already in the context. Avoid look in the cache loader.
         return null;
      }
      try {
         while (!cdl.lock(key, false));
         InternalCacheEntry ice = dataContainer.get(key);
         if (ice != null) {
            if (cmd instanceof ApplyDeltaCommand) {
               ctx.putLookedUpEntry(key, ice);
               entryFactory.wrapEntryForDelta(ctx, key, ((ApplyDeltaCommand) cmd).getDelta());
            } else {
               entryFactory.wrapEntryForPut(ctx, key, ice, false, cmd, false);
            }
            return null;
         }
         Boolean loaded = super.loadIfNeeded(ctx, key, isRetrieval, cmd);
         if (loaded == Boolean.TRUE) {
            if (enabled && isManualEviction) {
               // check if value was loaded
               CacheEntry e = ctx.lookupEntry(key);
               // We have to commit this before we remove from the store so there isn't a gap where the entry isn't visible
               // This should always be true given that loaded was true, but just sanity check
               if (e != null && e.isLoaded()) {
                  cdl.commitEntry(e, null, cmd, ctx);
                  removeFromStoreIfNeeded(key);
               }
            }
         }
         return loaded;
      } finally {
         cdl.unlock(key);
      }
   }

   @Override
   protected void sendNotification(Object key, Object value, boolean pre,
         InvocationContext ctx, FlagAffectedCommand cmd) {
      super.sendNotification(key, value, pre, ctx, cmd);
      notifier.notifyCacheEntryActivated(key, value, pre, ctx, cmd);
   }

   private void removeFromStoreIfNeeded(Object... keys) {
      if (enabled && isManualEviction) {
         if (log.isTraceEnabled())
            log.tracef("Remove from store keys=%s, if needed", Arrays.toString(keys));

         for (Object key: keys)
            activationManager.activate(key);
      }
   }

}

TOP

Related Classes of org.infinispan.interceptors.ActivationInterceptor

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.