Search in sources :

Example 1 with EvictVisitor

use of org.hibernate.event.internal.EvictVisitor in project hibernate-reactive by hibernate.

the class DefaultReactiveRefreshEventListener method reactiveOnRefresh.

private CompletionStage<Void> reactiveOnRefresh(RefreshEvent event, IdentitySet refreshedAlready, Object entity) {
    EventSource source = event.getSession();
    PersistenceContext persistenceContext = source.getPersistenceContextInternal();
    if (refreshedAlready.contains(entity)) {
        LOG.trace("Already refreshed");
        return voidFuture();
    }
    final EntityEntry e = persistenceContext.getEntry(entity);
    final EntityPersister persister;
    final Serializable id;
    if (e == null) {
        persister = source.getEntityPersister(event.getEntityName(), entity);
        // refresh() does not pass an entityName
        id = persister.getIdentifier(entity, event.getSession());
        if (LOG.isTraceEnabled()) {
            LOG.tracev("Refreshing transient {0}", infoString(persister, id, source.getFactory()));
        }
        if (persistenceContext.getEntry(source.generateEntityKey(id, persister)) != null) {
            throw new PersistentObjectException("attempted to refresh transient instance when persistent instance was already associated with the session: " + infoString(persister, id, source.getFactory()));
        }
    } else {
        if (LOG.isTraceEnabled()) {
            LOG.tracev("Refreshing ", infoString(e.getPersister(), e.getId(), source.getFactory()));
        }
        if (!e.isExistsInDatabase()) {
            throw new UnresolvableObjectException(e.getId(), "this instance does not yet exist as a row in the database");
        }
        persister = e.getPersister();
        id = e.getId();
    }
    // cascade the refresh prior to refreshing this entity
    refreshedAlready.add(entity);
    return cascadeRefresh(source, persister, entity, refreshedAlready).thenCompose(v -> {
        if (e != null) {
            final EntityKey key = source.generateEntityKey(id, persister);
            persistenceContext.removeEntity(key);
            if (persister.hasCollections()) {
                new EvictVisitor(source, entity).process(entity, persister);
            }
        }
        if (persister.canWriteToCache()) {
            Object previousVersion = null;
            if (persister.isVersionPropertyGenerated()) {
                // we need to grab the version value from the entity, otherwise
                // we have issues with generated-version entities that may have
                // multiple actions queued during the same flush
                previousVersion = persister.getVersion(entity);
            }
            final EntityDataAccess cache = persister.getCacheAccessStrategy();
            final Object ck = cache.generateCacheKey(id, persister, source.getFactory(), source.getTenantIdentifier());
            final SoftLock lock = cache.lockItem(source, ck, previousVersion);
            cache.remove(source, ck);
            source.getActionQueue().registerProcess((success, session) -> cache.unlockItem(session, ck, lock));
        }
        evictCachedCollections(persister, id, source);
        String previousFetchProfile = source.getLoadQueryInfluencers().getInternalFetchProfile();
        source.getLoadQueryInfluencers().setInternalFetchProfile("refresh");
        // Handle the requested lock-mode (if one) in relation to the entry's (if one) current lock-mode
        LockOptions lockOptionsToUse = event.getLockOptions();
        final LockMode requestedLockMode = lockOptionsToUse.getLockMode();
        final LockMode postRefreshLockMode;
        if (e != null) {
            final LockMode currentLockMode = e.getLockMode();
            if (currentLockMode.greaterThan(requestedLockMode)) {
                // the requested lock-mode is less restrictive than the current one
                // - pass along the current lock-mode (after accounting for WRITE)
                lockOptionsToUse = LockOptions.copy(event.getLockOptions(), new LockOptions());
                if (currentLockMode == LockMode.WRITE || currentLockMode == LockMode.PESSIMISTIC_WRITE || currentLockMode == LockMode.PESSIMISTIC_READ) {
                    // our transaction should already hold the exclusive lock on
                    // the underlying row - so READ should be sufficient.
                    // 
                    // in fact, this really holds true for any current lock-mode that indicates we
                    // hold an exclusive lock on the underlying row - but we *need* to handle
                    // WRITE specially because the Loader/Locker mechanism does not allow for WRITE
                    // locks
                    lockOptionsToUse.setLockMode(LockMode.READ);
                    // and prepare to reset the entry lock-mode to the previous lock mode after
                    // the refresh completes
                    postRefreshLockMode = currentLockMode;
                } else {
                    lockOptionsToUse.setLockMode(currentLockMode);
                    postRefreshLockMode = null;
                }
            } else {
                postRefreshLockMode = null;
            }
        } else {
            postRefreshLockMode = null;
        }
        return ((ReactiveAbstractEntityPersister) persister).reactiveLoad(id, entity, lockOptionsToUse, source).thenAccept(result -> {
            if (result != null) {
                // apply `postRefreshLockMode`, if needed
                if (postRefreshLockMode != null) {
                    // if we get here, there was a previous entry and we need to re-set its lock-mode
                    // - however, the refresh operation actually creates a new entry, so get it
                    persistenceContext.getEntry(result).setLockMode(postRefreshLockMode);
                }
                // If it was transient, then set it to the default for the source.
                if (!persister.isMutable()) {
                    // this is probably redundant; it should already be read-only
                    source.setReadOnly(result, true);
                } else {
                    source.setReadOnly(result, e == null ? source.isDefaultReadOnly() : e.isReadOnly());
                }
            }
            UnresolvableObjectException.throwIfNull(result, id, persister.getEntityName());
        }).whenComplete((vv, t) -> source.getLoadQueryInfluencers().setInternalFetchProfile(previousFetchProfile));
    });
}
Also used : EntityPersister(org.hibernate.persister.entity.EntityPersister) ReactiveAbstractEntityPersister(org.hibernate.reactive.persister.entity.impl.ReactiveAbstractEntityPersister) EntityPersister(org.hibernate.persister.entity.EntityPersister) EventSource(org.hibernate.event.spi.EventSource) CollectionType(org.hibernate.type.CollectionType) EntityDataAccess(org.hibernate.cache.spi.access.EntityDataAccess) Log(org.hibernate.reactive.logging.impl.Log) PersistenceContext(org.hibernate.engine.spi.PersistenceContext) UnresolvableObjectException(org.hibernate.UnresolvableObjectException) CollectionDataAccess(org.hibernate.cache.spi.access.CollectionDataAccess) LoggerFactory(org.hibernate.reactive.logging.impl.LoggerFactory) CompositeType(org.hibernate.type.CompositeType) IdentitySet(org.hibernate.internal.util.collections.IdentitySet) Map(java.util.Map) Cascade(org.hibernate.reactive.engine.impl.Cascade) CompletionStages.voidFuture(org.hibernate.reactive.util.impl.CompletionStages.voidFuture) SessionFactoryImplementor(org.hibernate.engine.spi.SessionFactoryImplementor) MetamodelImplementor(org.hibernate.metamodel.spi.MetamodelImplementor) ReactiveSession(org.hibernate.reactive.session.ReactiveSession) ActionQueue(org.hibernate.engine.spi.ActionQueue) LockMode(org.hibernate.LockMode) LockOptions(org.hibernate.LockOptions) MethodHandles(java.lang.invoke.MethodHandles) RefreshEvent(org.hibernate.event.spi.RefreshEvent) EntityKey(org.hibernate.engine.spi.EntityKey) Serializable(java.io.Serializable) RefreshEventListener(org.hibernate.event.spi.RefreshEventListener) MessageHelper.infoString(org.hibernate.pretty.MessageHelper.infoString) PersistentObjectException(org.hibernate.PersistentObjectException) SoftLock(org.hibernate.cache.spi.access.SoftLock) CascadingActions(org.hibernate.reactive.engine.impl.CascadingActions) CompletionStage(java.util.concurrent.CompletionStage) ReactiveRefreshEventListener(org.hibernate.reactive.event.ReactiveRefreshEventListener) EntityEntry(org.hibernate.engine.spi.EntityEntry) CascadePoint(org.hibernate.engine.internal.CascadePoint) EvictVisitor(org.hibernate.event.internal.EvictVisitor) HibernateException(org.hibernate.HibernateException) CollectionPersister(org.hibernate.persister.collection.CollectionPersister) ReactiveAbstractEntityPersister(org.hibernate.reactive.persister.entity.impl.ReactiveAbstractEntityPersister) Type(org.hibernate.type.Type) Serializable(java.io.Serializable) LockOptions(org.hibernate.LockOptions) PersistenceContext(org.hibernate.engine.spi.PersistenceContext) MessageHelper.infoString(org.hibernate.pretty.MessageHelper.infoString) LockMode(org.hibernate.LockMode) PersistentObjectException(org.hibernate.PersistentObjectException) ReactiveAbstractEntityPersister(org.hibernate.reactive.persister.entity.impl.ReactiveAbstractEntityPersister) EvictVisitor(org.hibernate.event.internal.EvictVisitor) EntityKey(org.hibernate.engine.spi.EntityKey) EventSource(org.hibernate.event.spi.EventSource) EntityEntry(org.hibernate.engine.spi.EntityEntry) UnresolvableObjectException(org.hibernate.UnresolvableObjectException) EntityDataAccess(org.hibernate.cache.spi.access.EntityDataAccess) SoftLock(org.hibernate.cache.spi.access.SoftLock)

Aggregations

Serializable (java.io.Serializable)1 MethodHandles (java.lang.invoke.MethodHandles)1 Map (java.util.Map)1 CompletionStage (java.util.concurrent.CompletionStage)1 HibernateException (org.hibernate.HibernateException)1 LockMode (org.hibernate.LockMode)1 LockOptions (org.hibernate.LockOptions)1 PersistentObjectException (org.hibernate.PersistentObjectException)1 UnresolvableObjectException (org.hibernate.UnresolvableObjectException)1 CollectionDataAccess (org.hibernate.cache.spi.access.CollectionDataAccess)1 EntityDataAccess (org.hibernate.cache.spi.access.EntityDataAccess)1 SoftLock (org.hibernate.cache.spi.access.SoftLock)1 CascadePoint (org.hibernate.engine.internal.CascadePoint)1 ActionQueue (org.hibernate.engine.spi.ActionQueue)1 EntityEntry (org.hibernate.engine.spi.EntityEntry)1 EntityKey (org.hibernate.engine.spi.EntityKey)1 PersistenceContext (org.hibernate.engine.spi.PersistenceContext)1 SessionFactoryImplementor (org.hibernate.engine.spi.SessionFactoryImplementor)1 EvictVisitor (org.hibernate.event.internal.EvictVisitor)1 EventSource (org.hibernate.event.spi.EventSource)1