From 1d62197b9dc20fb7770addd82be716a6e5a69939 Mon Sep 17 00:00:00 2001 From: Radim Vansa Date: Wed, 12 Aug 2015 13:09:53 +0200 Subject: [PATCH] HHH-10008 SessionImplementor.getTimestamp() does not return transaction start time; HHH-9962 Second level query cache returns stale data if query and update statements are executed concurrently --- .../cache/internal/StandardQueryCache.java | 6 +- .../org/hibernate/internal/SessionImpl.java | 4 + .../internal/StatelessSessionImpl.java | 4 + .../test/querycache/QueryCacheTest.java | 131 ++++++++++++++++++ 4 files changed, 141 insertions(+), 4 deletions(-) diff --git a/hibernate-core/src/main/java/org/hibernate/cache/internal/StandardQueryCache.java b/hibernate-core/src/main/java/org/hibernate/cache/internal/StandardQueryCache.java index 99171bf7c4..82f7cbb58f 100644 --- a/hibernate-core/src/main/java/org/hibernate/cache/internal/StandardQueryCache.java +++ b/hibernate-core/src/main/java/org/hibernate/cache/internal/StandardQueryCache.java @@ -111,15 +111,13 @@ public class StandardQueryCache implements QueryCache { if ( isNaturalKeyLookup && result.isEmpty() ) { return false; } - final long ts = cacheRegion.nextTimestamp(); - if ( DEBUGGING ) { - LOG.debugf( "Caching query results in region: %s; timestamp=%s", cacheRegion.getName(), ts ); + LOG.debugf( "Caching query results in region: %s; timestamp=%s", cacheRegion.getName(), session.getTimestamp() ); } final List cacheable = new ArrayList( result.size() + 1 ); logCachedResultDetails( key, null, returnTypes, cacheable ); - cacheable.add( ts ); + cacheable.add( session.getTimestamp() ); final boolean isSingleResult = returnTypes.length == 1; for ( Object aResult : result ) { diff --git a/hibernate-core/src/main/java/org/hibernate/internal/SessionImpl.java b/hibernate-core/src/main/java/org/hibernate/internal/SessionImpl.java index a5888f9135..1d481498e7 100644 --- a/hibernate-core/src/main/java/org/hibernate/internal/SessionImpl.java +++ b/hibernate-core/src/main/java/org/hibernate/internal/SessionImpl.java @@ -1436,6 +1436,10 @@ public final class SessionImpl extends AbstractSessionImpl implements EventSourc public Transaction beginTransaction() throws HibernateException { errorIfClosed(); Transaction result = getTransaction(); + // begin on already started transaction is noop, therefore, don't update the timestamp + if (result.getStatus() != TransactionStatus.ACTIVE) { + timestamp = factory.getSettings().getRegionFactory().nextTimestamp(); + } result.begin(); return result; } diff --git a/hibernate-core/src/main/java/org/hibernate/internal/StatelessSessionImpl.java b/hibernate-core/src/main/java/org/hibernate/internal/StatelessSessionImpl.java index 9c2c425072..73d4dde887 100755 --- a/hibernate-core/src/main/java/org/hibernate/internal/StatelessSessionImpl.java +++ b/hibernate-core/src/main/java/org/hibernate/internal/StatelessSessionImpl.java @@ -528,6 +528,10 @@ public class StatelessSessionImpl extends AbstractSessionImpl implements Statele public Transaction beginTransaction() throws HibernateException { errorIfClosed(); Transaction result = getTransaction(); + // begin on already started transaction is noop, therefore, don't update the timestamp + if (result.getStatus() != TransactionStatus.ACTIVE) { + timestamp = factory.getSettings().getRegionFactory().nextTimestamp(); + } result.begin(); return result; } diff --git a/hibernate-core/src/test/java/org/hibernate/test/querycache/QueryCacheTest.java b/hibernate-core/src/test/java/org/hibernate/test/querycache/QueryCacheTest.java index 9c96b3be7f..a2bcb49a5a 100755 --- a/hibernate-core/src/test/java/org/hibernate/test/querycache/QueryCacheTest.java +++ b/hibernate-core/src/test/java/org/hibernate/test/querycache/QueryCacheTest.java @@ -6,15 +6,24 @@ */ package org.hibernate.test.querycache; +import java.io.Serializable; import java.util.ArrayList; import java.util.List; import java.util.Map; +import java.util.concurrent.Callable; +import java.util.concurrent.CountDownLatch; +import java.util.concurrent.ExecutionException; +import java.util.concurrent.ExecutorService; +import java.util.concurrent.Executors; +import java.util.concurrent.Future; import org.hibernate.Criteria; +import org.hibernate.EmptyInterceptor; import org.hibernate.Hibernate; import org.hibernate.Query; import org.hibernate.SQLQuery; import org.hibernate.Session; +import org.hibernate.SessionBuilder; import org.hibernate.Transaction; import org.hibernate.cfg.AvailableSettings; import org.hibernate.criterion.Restrictions; @@ -26,6 +35,7 @@ import org.hibernate.testing.DialectChecks; import org.hibernate.testing.RequiresDialectFeature; import org.hibernate.testing.TestForIssue; import org.hibernate.testing.junit4.BaseNonConfigCoreFunctionalTestCase; +import org.hibernate.type.Type; import org.junit.Test; import static org.junit.Assert.assertEquals; @@ -39,6 +49,7 @@ import static org.junit.Assert.assertTrue; public class QueryCacheTest extends BaseNonConfigCoreFunctionalTestCase { private static final CompositeKey PK = new CompositeKey(1, 2); + private static final ExecutorService executor = Executors.newFixedThreadPool(4); @Override public String[] getMappings() { @@ -63,6 +74,17 @@ public class QueryCacheTest extends BaseNonConfigCoreFunctionalTestCase { settings.put( AvailableSettings.GENERATE_STATISTICS, "true" ); } + @Override + protected void shutDown() { + super.shutDown(); + executor.shutdown(); + } + + @Override + protected boolean isCleanupTestDataRequired() { + return true; + } + @Override protected String getCacheConcurrencyStrategy() { return "nonstrict-read-write"; @@ -529,5 +551,114 @@ public class QueryCacheTest extends BaseNonConfigCoreFunctionalTestCase { // assertEquals(1, query.getResultList().size()); // } + @Test + @TestForIssue(jiraKey = "HHH-9962") + /* Test courtesy of Giambattista Bloisi */ + public void testDelayedLoad() throws InterruptedException, ExecutionException { + DelayLoadOperations interceptor = new DelayLoadOperations(); + final SessionBuilder sessionBuilder = sessionFactory().withOptions().interceptor(interceptor); + Item item1 = new Item(); + item1.setName("Item1"); + item1.setDescription("Washington"); + Session s1 = sessionBuilder.openSession(); + Transaction tx1 = s1.beginTransaction(); + s1.persist(item1); + tx1.commit(); + s1.close(); + + Item item2 = new Item(); + item2.setName("Item2"); + item2.setDescription("Chicago"); + Session s2 = sessionBuilder.openSession(); + Transaction tx2 = s2.beginTransaction(); + s2.persist(item2); + tx2.commit(); + s2.close(); + + interceptor.blockOnLoad(); + + Future fetchedItem = executor.submit(new Callable() { + public Item call() throws Exception { + return findByDescription(sessionBuilder, "Washington"); + } + }); + + // wait for the onLoad listener to be called + interceptor.waitOnLoad(); + + Session s3 = sessionBuilder.openSession(); + Transaction tx3 = s3.beginTransaction(); + item1.setDescription("New York"); + item2.setDescription("Washington"); + s3.update(item1); + s3.update(item2); + tx3.commit(); + s3.close(); + + interceptor.unblockOnLoad(); + + // the concurrent query was executed before the data was amended so + // let's expect "Item1" to be returned as living in Washington + Item fetched = fetchedItem.get(); + assertEquals("Item1", fetched.getName()); + + // Query again: now "Item2" is expected to live in Washington + fetched = findByDescription(sessionBuilder, "Washington"); + assertEquals("Item2", fetched.getName()); + } + + protected Item findByDescription(SessionBuilder sessionBuilder, final String description) { + Session s = sessionBuilder.openSession(); + try { + return (Item) s.createCriteria(Item.class) + .setCacheable(true) + .setReadOnly(true) + .add(Restrictions.eq("description", description)) + .uniqueResult(); + + } finally { + s.close(); + } + } + + public class DelayLoadOperations extends EmptyInterceptor { + + private volatile CountDownLatch blockLatch; + private volatile CountDownLatch waitLatch; + + @Override + public boolean onLoad(Object entity, Serializable id, Object[] state, String[] propertyNames, Type[] types) { + // Synchronize load and update activities + try { + if (waitLatch != null) { + waitLatch.countDown(); + waitLatch = null; + } + if (blockLatch != null) { + blockLatch.await(); + blockLatch = null; + } + } catch (InterruptedException e) { + Thread.currentThread().interrupt(); + throw new RuntimeException(e); + } + return true; + } + + public void blockOnLoad() { + blockLatch = new CountDownLatch(1); + waitLatch = new CountDownLatch(1); + } + + public void waitOnLoad() throws InterruptedException { + waitLatch.await(); + } + + public void unblockOnLoad() { + if (blockLatch != null) { + blockLatch.countDown(); + } + } + } }