/* * Hibernate, Relational Persistence for Idiomatic Java * * Copyright (c) 2008, Red Hat Middleware LLC or third-party contributors as * indicated by the @author tags or express copyright attribution * statements applied by the authors. All third-party contributions are * distributed under license by Red Hat Middleware LLC. * * This copyrighted material is made available to anyone wishing to use, modify, * copy, or redistribute it subject to the terms and conditions of the GNU * Lesser General Public License, as published by the Free Software Foundation. * * This program 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 distribution; if not, write to: * Free Software Foundation, Inc. * 51 Franklin Street, Fifth Floor * Boston, MA 02110-1301 USA * */ package org.hibernate.action; import org.hibernate.cache.access.SoftLock; import org.hibernate.cache.CacheException; import org.hibernate.cache.CacheKey; import org.hibernate.collection.PersistentCollection; import org.hibernate.engine.SessionImplementor; import org.hibernate.persister.collection.CollectionPersister; import org.hibernate.pretty.MessageHelper; import org.hibernate.util.StringHelper; import java.io.IOException; import java.io.ObjectInputStream; import java.io.Serializable; /** * Any action relating to insert/update/delete of a collection * * @author Gavin King */ public abstract class CollectionAction implements Executable, Serializable, Comparable { private transient CollectionPersister persister; private final Serializable key; private final SessionImplementor session; private final String collectionRole; private final PersistentCollection collection; public CollectionAction( final CollectionPersister persister, final PersistentCollection collection, final Serializable key, final SessionImplementor session) throws CacheException { this.persister = persister; this.session = session; this.key = key; this.collectionRole = persister.getRole(); this.collection = collection; } protected PersistentCollection getCollection() { return collection; } private void readObject(ObjectInputStream ois) throws IOException, ClassNotFoundException { ois.defaultReadObject(); persister = session.getFactory().getCollectionPersister( collectionRole ); } public final void beforeExecutions() throws CacheException { // we need to obtain the lock before any actions are // executed, since this may be an inverse="true" // bidirectional association and it is one of the // earlier entity actions which actually updates // the database (this action is resposible for // second-level cache invalidation only) if ( persister.hasCache() ) { final CacheKey ck = new CacheKey( key, persister.getKeyType(), persister.getRole(), session.getEntityMode(), session.getFactory() ); final SoftLock lock = persister.getCacheAccessStrategy().lockItem( ck, null ); // the old behavior used key as opposed to getKey() afterTransactionProcess = new CacheCleanupProcess( key, persister, lock ); } } public BeforeTransactionCompletionProcess getBeforeTransactionCompletionProcess() { return null; } private AfterTransactionCompletionProcess afterTransactionProcess; public AfterTransactionCompletionProcess getAfterTransactionCompletionProcess() { return afterTransactionProcess; } public Serializable[] getPropertySpaces() { return persister.getCollectionSpaces(); } protected final CollectionPersister getPersister() { return persister; } protected final Serializable getKey() { Serializable finalKey = key; if ( key instanceof DelayedPostInsertIdentifier ) { // need to look it up from the persistence-context finalKey = session.getPersistenceContext().getEntry( collection.getOwner() ).getId(); if ( finalKey == key ) { // we may be screwed here since the collection action is about to execute // and we do not know the final owner key value } } return finalKey; } protected final SessionImplementor getSession() { return session; } protected final void evict() throws CacheException { if ( persister.hasCache() ) { CacheKey ck = new CacheKey( key, persister.getKeyType(), persister.getRole(), session.getEntityMode(), session.getFactory() ); persister.getCacheAccessStrategy().remove( ck ); } } public String toString() { return StringHelper.unqualify( getClass().getName() ) + MessageHelper.infoString( collectionRole, key ); } public int compareTo(Object other) { CollectionAction action = ( CollectionAction ) other; //sort first by role name int roleComparison = collectionRole.compareTo( action.collectionRole ); if ( roleComparison != 0 ) { return roleComparison; } else { //then by fk return persister.getKeyType() .compare( key, action.key, session.getEntityMode() ); } } private static class CacheCleanupProcess implements AfterTransactionCompletionProcess { private final Serializable key; private final CollectionPersister persister; private final SoftLock lock; private CacheCleanupProcess(Serializable key, CollectionPersister persister, SoftLock lock) { this.key = key; this.persister = persister; this.lock = lock; } public void doAfterTransactionCompletion(boolean success, SessionImplementor session) { final CacheKey ck = new CacheKey( key, persister.getKeyType(), persister.getRole(), session.getEntityMode(), session.getFactory() ); persister.getCacheAccessStrategy().unlockItem( ck, lock ); } } }