All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.hibernate.cache.infinispan.impl.BaseRegion Maven / Gradle / Ivy

There is a newer version: 5.6.15.Final
Show newest version
/*
 * Hibernate, Relational Persistence for Idiomatic Java
 *
 * License: GNU Lesser General Public License (LGPL), version 2.1 or later.
 * See the lgpl.txt file in the root directory or .
 */
package org.hibernate.cache.infinispan.impl;

import java.util.Collections;
import java.util.Map;
import java.util.concurrent.atomic.AtomicReference;
import javax.transaction.SystemException;
import javax.transaction.Transaction;
import javax.transaction.TransactionManager;

import org.hibernate.cache.CacheException;
import org.hibernate.cache.infinispan.access.PutFromLoadValidator;
import org.hibernate.cache.infinispan.util.Caches;
import org.hibernate.cache.spi.Region;
import org.hibernate.cache.spi.RegionFactory;

import org.hibernate.cache.spi.access.AccessType;
import org.infinispan.AdvancedCache;
import org.infinispan.context.Flag;
import org.infinispan.util.logging.Log;
import org.infinispan.util.logging.LogFactory;

/**
 * Support for Infinispan {@link Region}s. Handles common "utility" methods for an underlying named
 * Cache. In other words, this implementation doesn't actually read or write data. Subclasses are
 * expected to provide core cache interaction appropriate to the semantics needed.
 *
 * @author Chris Bredesen
 * @author Galder Zamarreño
 * @since 3.5
 */
public abstract class BaseRegion implements Region {

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

	private enum InvalidateState {
		INVALID, CLEARING, VALID
	}

	private final String name;
	private final AdvancedCache localAndSkipLoadCache;
	private final TransactionManager tm;

	private final Object invalidationMutex = new Object();
	private final AtomicReference invalidateState =
			new AtomicReference( InvalidateState.VALID );

	private final RegionFactory factory;

	protected final AdvancedCache cache;

	private PutFromLoadValidator validator;

   /**
    * Base region constructor.
    *
    * @param cache instance for the region
    * @param name of the region
	 * @param transactionManager transaction manager may be needed even for non-transactional caches.
    * @param factory for this region
    */
	public BaseRegion(AdvancedCache cache, String name, TransactionManager transactionManager, RegionFactory factory) {
		this.cache = cache;
		this.name = name;
		this.tm = transactionManager;
		this.factory = factory;
		this.localAndSkipLoadCache = cache.withFlags(
				Flag.CACHE_MODE_LOCAL, Flag.ZERO_LOCK_ACQUISITION_TIMEOUT,
				Flag.SKIP_CACHE_LOAD
		);
	}

	@Override
	public String getName() {
		return name;
	}

	@Override
	public long getElementCountInMemory() {
		if ( checkValid() ) {
			return localAndSkipLoadCache.size();
		}

		return 0;
	}

	/**
	 * {@inheritDoc}
	 * 

* Not supported; returns -1 */ @Override public long getElementCountOnDisk() { return -1; } /** * {@inheritDoc} *

* Not supported; returns -1 */ @Override public long getSizeInMemory() { return -1; } @Override public int getTimeout() { // 60 seconds return 60000; } @Override public long nextTimestamp() { return factory.nextTimestamp(); } @Override public Map toMap() { if ( checkValid() ) { return cache; } return Collections.EMPTY_MAP; } @Override public void destroy() throws CacheException { cache.stop(); } @Override public boolean contains(Object key) { return checkValid() && cache.containsKey( key ); } /** * Checks if the region is valid for operations such as storing new data * in the region, or retrieving data from the region. * * @return true if the region is valid, false otherwise */ public boolean checkValid() { boolean valid = isValid(); if ( !valid ) { synchronized (invalidationMutex) { if ( invalidateState.compareAndSet( InvalidateState.INVALID, InvalidateState.CLEARING ) ) { try { runInvalidation( getCurrentTransaction() != null ); log.tracef( "Transition state from CLEARING to VALID" ); invalidateState.compareAndSet( InvalidateState.CLEARING, InvalidateState.VALID ); } catch ( Exception e ) { if ( log.isTraceEnabled() ) { log.trace( "Could not invalidate region: ", e ); } } } } valid = isValid(); } return valid; } protected boolean isValid() { return invalidateState.get() == InvalidateState.VALID; } /** * Tell the TransactionManager to suspend any ongoing transaction. * * @return the transaction that was suspended, or null if * there wasn't one */ public Transaction suspend() { Transaction tx = null; try { if ( tm != null ) { tx = tm.suspend(); } } catch (SystemException se) { throw new CacheException( "Could not suspend transaction", se ); } return tx; } /** * Tell the TransactionManager to resume the given transaction * * @param tx the transaction to suspend. May be null. */ public void resume(Transaction tx) { try { if ( tx != null ) { tm.resume( tx ); } } catch (Exception e) { throw new CacheException( "Could not resume transaction", e ); } } /** * Invalidates the region. */ public void invalidateRegion() { if (log.isTraceEnabled()) { log.trace( "Invalidate region: " + name ); } invalidateState.set(InvalidateState.INVALID); } public TransactionManager getTransactionManager() { return tm; } // Used to satisfy TransactionalDataRegion.isTransactionAware in subclasses @SuppressWarnings("unused") public boolean isTransactionAware() { return tm != null; } public AdvancedCache getCache() { return cache; } private Transaction getCurrentTransaction() { try { // Transaction manager could be null return tm != null ? tm.getTransaction() : null; } catch (SystemException e) { throw new CacheException("Unable to get current transaction", e); } } protected void checkAccessType(AccessType accessType) { if (accessType == AccessType.TRANSACTIONAL && !cache.getCacheConfiguration().transaction().transactionMode().isTransactional()) { log.warn("Requesting TRANSACTIONAL cache concurrency strategy but the cache is not configured as transactional."); } else if (accessType == AccessType.READ_WRITE && cache.getCacheConfiguration().transaction().transactionMode().isTransactional()) { log.warn("Requesting READ_WRITE cache concurrency strategy but the cache was configured as transactional."); } } protected synchronized PutFromLoadValidator getValidator() { if (validator == null) { validator = new PutFromLoadValidator(cache); } return validator; } protected void runInvalidation(boolean inTransaction) { // If we're running inside a transaction, we need to remove elements one-by-one // to clean the context as well (cache.clear() does not do that). // When we don't have transaction, we can do a clear operation (since we don't // case about context) and can't do the one-by-one remove: remove() on tx cache // requires transactional context. if ( inTransaction ) { log.tracef( "Transaction, clearing one element at the time" ); Caches.removeAll( localAndSkipLoadCache ); } else { log.tracef( "Non-transactional, clear in one go" ); localAndSkipLoadCache.clear(); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy