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

org.hibernate.action.internal.BulkOperationCleanupAction Maven / Gradle / Ivy

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

import java.io.Serializable;
import java.util.Arrays;
import java.util.HashSet;
import java.util.LinkedHashSet;
import java.util.Set;

import org.hibernate.HibernateException;
import org.hibernate.action.spi.AfterTransactionCompletionProcess;
import org.hibernate.action.spi.BeforeTransactionCompletionProcess;
import org.hibernate.action.spi.Executable;
import org.hibernate.cache.spi.access.CollectionDataAccess;
import org.hibernate.cache.spi.access.EntityDataAccess;
import org.hibernate.cache.spi.access.NaturalIdDataAccess;
import org.hibernate.cache.spi.access.SoftLock;
import org.hibernate.engine.spi.SessionFactoryImplementor;
import org.hibernate.engine.spi.SharedSessionContractImplementor;
import org.hibernate.metamodel.spi.MetamodelImplementor;
import org.hibernate.persister.collection.CollectionPersister;
import org.hibernate.persister.entity.EntityPersister;
import org.hibernate.persister.entity.Queryable;

/**
 * An {@link org.hibernate.engine.spi.ActionQueue} {@link org.hibernate.action.spi.Executable} for ensuring
 * shared cache cleanup in relation to performed bulk HQL queries.
 * 

* NOTE: currently this executes for INSERT queries as well as * UPDATE and DELETE queries. For INSERT it is * really not needed as we'd have no invalid entity/collection data to * cleanup (we'd still nee to invalidate the appropriate update-timestamps * regions) as a result of this query. * * @author Steve Ebersole */ public class BulkOperationCleanupAction implements Executable, Serializable { private final Serializable[] affectedTableSpaces; private final Set entityCleanups = new HashSet<>(); private final Set collectionCleanups = new HashSet<>(); private final Set naturalIdCleanups = new HashSet<>(); /** * Constructs an action to cleanup "affected cache regions" based on the * affected entity persisters. The affected regions are defined as the * region (if any) of the entity persisters themselves, plus the * collection regions for any collection in which those entity * persisters participate as elements/keys/etc. * * @param session The session to which this request is tied. * @param affectedQueryables The affected entity persisters. */ public BulkOperationCleanupAction(SharedSessionContractImplementor session, Queryable... affectedQueryables) { final SessionFactoryImplementor factory = session.getFactory(); final LinkedHashSet spacesList = new LinkedHashSet<>(); for ( Queryable persister : affectedQueryables ) { spacesList.addAll( Arrays.asList( (String[]) persister.getQuerySpaces() ) ); if ( persister.canWriteToCache() ) { final EntityDataAccess entityDataAccess = persister.getCacheAccessStrategy(); if ( entityDataAccess != null ) { entityCleanups.add( new EntityCleanup( entityDataAccess, session ) ); } } if ( persister.hasNaturalIdentifier() && persister.hasNaturalIdCache() ) { naturalIdCleanups.add( new NaturalIdCleanup( persister.getNaturalIdCacheAccessStrategy(), session ) ); } final Set roles = factory.getMetamodel().getCollectionRolesByEntityParticipant( persister.getEntityName() ); if ( roles != null ) { for ( String role : roles ) { final CollectionPersister collectionPersister = factory.getMetamodel().collectionPersister( role ); if ( collectionPersister.hasCache() ) { collectionCleanups.add( new CollectionCleanup( collectionPersister.getCacheAccessStrategy(), session ) ); } } } } this.affectedTableSpaces = spacesList.toArray( new String[ 0 ] ); } /** * Constructs an action to cleanup "affected cache regions" based on a * set of affected table spaces. This differs from {@link #BulkOperationCleanupAction(SharedSessionContractImplementor, Queryable[])} * in that here we have the affected table names. From those * we deduce the entity persisters which are affected based on the defined * {@link EntityPersister#getQuerySpaces() table spaces}; and from there, we * determine the affected collection regions based on any collections * in which those entity persisters participate as elements/keys/etc. * * @param session The session to which this request is tied. * @param tableSpaces The table spaces. */ @SuppressWarnings( { "unchecked", "rawtypes" } ) public BulkOperationCleanupAction(SharedSessionContractImplementor session, Set tableSpaces) { final LinkedHashSet spacesList = new LinkedHashSet<>( tableSpaces ); final SessionFactoryImplementor factory = session.getFactory(); final MetamodelImplementor metamodel = factory.getMetamodel(); for ( EntityPersister persister : metamodel.entityPersisters().values() ) { final String[] entitySpaces = (String[]) persister.getQuerySpaces(); if ( affectedEntity( tableSpaces, entitySpaces ) ) { spacesList.addAll( Arrays.asList( entitySpaces ) ); if ( persister.canWriteToCache() ) { entityCleanups.add( new EntityCleanup( persister.getCacheAccessStrategy(), session ) ); } if ( persister.hasNaturalIdentifier() && persister.hasNaturalIdCache() ) { naturalIdCleanups.add( new NaturalIdCleanup( persister.getNaturalIdCacheAccessStrategy(), session ) ); } final Set roles = metamodel.getCollectionRolesByEntityParticipant( persister.getEntityName() ); if ( roles != null ) { for ( String role : roles ) { final CollectionPersister collectionPersister = metamodel.collectionPersister( role ); if ( collectionPersister.hasCache() ) { collectionCleanups.add( new CollectionCleanup( collectionPersister.getCacheAccessStrategy(), session ) ); } } } } } this.affectedTableSpaces = spacesList.toArray( new String[ 0 ] ); } /** * Check whether we should consider an entity as affected by the query. This * defines inclusion of the entity in the clean-up. * * @param affectedTableSpaces The table spaces reported to be affected by * the query. * @param checkTableSpaces The table spaces (from the entity persister) * to check against the affected table spaces. * * @return Whether the entity should be considered affected * * @implNote An entity is considered to be affected if either (1) the affected table * spaces are not known or (2) any of the incoming check table spaces occur * in that set. */ private boolean affectedEntity(Set affectedTableSpaces, Serializable[] checkTableSpaces) { if ( affectedTableSpaces == null || affectedTableSpaces.isEmpty() ) { return true; } for ( Serializable checkTableSpace : checkTableSpaces ) { if ( affectedTableSpaces.contains( checkTableSpace ) ) { return true; } } return false; } @Override public Serializable[] getPropertySpaces() { return affectedTableSpaces; } @Override public BeforeTransactionCompletionProcess getBeforeTransactionCompletionProcess() { return null; } @Override public AfterTransactionCompletionProcess getAfterTransactionCompletionProcess() { return (success, session) -> { for ( EntityCleanup cleanup : entityCleanups ) { cleanup.release(); } entityCleanups.clear(); for ( NaturalIdCleanup cleanup : naturalIdCleanups ) { cleanup.release(); } naturalIdCleanups.clear(); for ( CollectionCleanup cleanup : collectionCleanups ) { cleanup.release(); } collectionCleanups.clear(); }; } @Override public void beforeExecutions() throws HibernateException { // nothing to do } @Override public void execute() throws HibernateException { // nothing to do } private static class EntityCleanup implements Serializable { private final EntityDataAccess cacheAccess; private final SoftLock cacheLock; private EntityCleanup( EntityDataAccess cacheAccess, SharedSessionContractImplementor session) { this.cacheAccess = cacheAccess; this.cacheLock = cacheAccess.lockRegion(); cacheAccess.removeAll( session ); } private void release() { cacheAccess.unlockRegion( cacheLock ); } } private static class CollectionCleanup implements Serializable { private final CollectionDataAccess cacheAccess; private final SoftLock cacheLock; private CollectionCleanup( CollectionDataAccess cacheAccess, SharedSessionContractImplementor session) { this.cacheAccess = cacheAccess; this.cacheLock = cacheAccess.lockRegion(); cacheAccess.removeAll( session ); } private void release() { cacheAccess.unlockRegion( cacheLock ); } } private static class NaturalIdCleanup implements Serializable { private final NaturalIdDataAccess naturalIdCacheAccessStrategy; private final SoftLock cacheLock; public NaturalIdCleanup( NaturalIdDataAccess naturalIdCacheAccessStrategy, SharedSessionContractImplementor session) { this.naturalIdCacheAccessStrategy = naturalIdCacheAccessStrategy; this.cacheLock = naturalIdCacheAccessStrategy.lockRegion(); naturalIdCacheAccessStrategy.removeAll( session ); } private void release() { naturalIdCacheAccessStrategy.unlockRegion( cacheLock ); } } @Override public void afterDeserialize(SharedSessionContractImplementor session) { // nop } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy