[ 
https://issues.apache.org/jira/browse/GEODE-7535?page=com.atlassian.jira.plugin.system.issuetabpanels:all-tabpanel
 ]

Ernest Burghardt closed GEODE-7535.
-----------------------------------
    Assignee: Ernest Burghardt  (was: Juan Ramos)

> Race Condition in LocalRegion.getAll
> ------------------------------------
>
>                 Key: GEODE-7535
>                 URL: https://issues.apache.org/jira/browse/GEODE-7535
>             Project: Geode
>          Issue Type: Bug
>          Components: client/server, regions
>            Reporter: Juan Ramos
>            Assignee: Ernest Burghardt
>            Priority: Major
>              Labels: GeodeCommons
>             Fix For: 1.12.0
>
>          Time Spent: 20m
>  Remaining Estimate: 0h
>
> The issue only happens for client regions with local storage 
> ({{CACHING_PROXY}} or similar), and in the absence of transactions.
>  The {{getAll}} operation first retrieves the entry using the {{accessEntry}} 
> method and, if the result is not {{null}}, it uses {{NonTxEntry.getValue()}} 
> to obtain the actual value and add it to the result {{Map}}. The method 
> {{NonTxEntry.getValue()}}, in turn, ends up callingĀ 
> {{region.basicGetEntry(key)}} again.
> {code:java|title=LocalRegion.java|borderStyle=solid}
> // Some comments here
> Map basicGetAll(Collection keys, Object callback) {
> ...
>       if (hasServerProxy()) {
>               ...
>               // Gather any local values
>               // We only need to do this if this region has local storage
>               if (getTXState() == null && hasStorage()) {
>                       ...
>                       for (Iterator iterator = keysList.iterator(); 
> iterator.hasNext();) {
>                               Object key = iterator.next();
>                               Object value;
>                               Region.Entry entry = accessEntry(key, true);
>                               if (entry != null && (value = entry.getValue()) 
> != null) {
>                                       allResults.put(key, value);
>                                       iterator.remove();
>                               }
>                       }
>               }
> ...
> {code}
> {code:java|title=NonTxEntry.java|borderStyle=solid}
> private RegionEntry basicGetEntry() {
>       RegionEntry re = region.basicGetEntry(key);
>       if (re == null) {
>               throw new EntryDestroyedException(key.toString());
>       }
>       return re;
>  }
> {code}
> If the entry is locally destroyed between the two retrievals ({{accessEntry}} 
> and {{basicGetEntry}}), which is rare but can happen as a consequence of 
> eviction/expiration as an example, the {{getAll}} operation will fail and the 
> following exception will be returned to the caller:
> {noformat}
> [vm2] [info 2019/12/04 15:51:52.949 GMT <RMI TCP Connection(1)-127.0.0.1> 
> tid=0x13] Got result: EXCEPTION_OCCURRED
> [vm2] org.apache.geode.cache.EntryDestroyedException: Key1
> [vm2]         at 
> org.apache.geode.internal.cache.NonTXEntry.basicGetEntry(NonTXEntry.java:62)
> [vm2]         at 
> org.apache.geode.internal.cache.NonTXEntry.getValue(NonTXEntry.java:91)
> [vm2]         at 
> org.apache.geode.internal.cache.NonTXEntry.getValue(NonTXEntry.java:86)
> [vm2]         at 
> org.apache.geode.internal.cache.LocalRegion.basicGetAll(LocalRegion.java:8611)
> [vm2]         at 
> org.apache.geode.internal.cache.AbstractRegion.getAll(AbstractRegion.java:530)
> [vm2]         at 
> org.apache.geode.internal.cache.AbstractRegion.getAll(AbstractRegion.java:520)
> [vm2]         at 
> org.apache.geode.internal.cache.ClientCachingProxyRegionDistributedTest.lambda$getAllShouldNotThrowExceptionWhenEntryIsLocallyDeletedBetweenFetches$bb17a952$1(ClientCachingProxyRegionDistributedTest.java:102)
> [vm2]         at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
> [vm2]         at 
> sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
> [vm2]         at 
> sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
> [vm2]         at java.lang.reflect.Method.invoke(Method.java:498)
> [vm2]         at 
> org.apache.geode.test.dunit.internal.MethodInvoker.executeObject(MethodInvoker.java:123)
> [vm2]         at 
> org.apache.geode.test.dunit.internal.RemoteDUnitVM.executeMethodOnObject(RemoteDUnitVM.java:78)
> [vm2]         at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
> [vm2]         at 
> sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
> [vm2]         at 
> sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
> [vm2]         at java.lang.reflect.Method.invoke(Method.java:498)
> [vm2]         at 
> sun.rmi.server.UnicastServerRef.dispatch(UnicastServerRef.java:357)
> [vm2]         at sun.rmi.transport.Transport$1.run(Transport.java:200)
> [vm2]         at sun.rmi.transport.Transport$1.run(Transport.java:197)
> [vm2]         at java.security.AccessController.doPrivileged(Native Method)
> [vm2]         at sun.rmi.transport.Transport.serviceCall(Transport.java:196)
> [vm2]         at 
> sun.rmi.transport.tcp.TCPTransport.handleMessages(TCPTransport.java:573)
> [vm2]         at 
> sun.rmi.transport.tcp.TCPTransport$ConnectionHandler.run0(TCPTransport.java:834)
> [vm2]         at 
> sun.rmi.transport.tcp.TCPTransport$ConnectionHandler.lambda$run$0(TCPTransport.java:688)
> [vm2]         at java.security.AccessController.doPrivileged(Native Method)
> [vm2]         at 
> sun.rmi.transport.tcp.TCPTransport$ConnectionHandler.run(TCPTransport.java:687)
> [vm2]         at 
> java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1149)
> [vm2]         at 
> java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:624)
> [vm2]         at java.lang.Thread.run(Thread.java:748)
> [vm2]  from 
> org.apache.geode.internal.cache.ClientCachingProxyRegionDistributedTest$$Lambda$135/2143459709.run
>  with 0 args on object: 
> org.apache.geode.internal.cache.ClientCachingProxyRegionDistributedTest$$Lambda$135/2143459709@7d3c7d1c
>  (took 929 ms)
> {noformat}
> The {{getAll}} operation should not throw this {{Exception}}; it should, 
> instead, retry internally and retrieve the entry from the server as it 
> usually does for non locally existing entries.



--
This message was sent by Atlassian Jira
(v8.3.4#803005)

Reply via email to