<div dir="ltr"><br><br><div class="gmail_quote"><div dir="ltr">On Tue, Mar 21, 2017 at 1:42 PM William Burns <<a href="mailto:mudokonman@gmail.com">mudokonman@gmail.com</a>> wrote:<br></div><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex"><div dir="ltr" class="gmail_msg"><div class="gmail_quote gmail_msg"><div dir="ltr" class="gmail_msg">On Tue, Mar 21, 2017 at 12:53 PM Radim Vansa <<a href="mailto:rvansa@redhat.com" class="gmail_msg" target="_blank">rvansa@redhat.com</a>> wrote:<br class="gmail_msg"></div><blockquote class="gmail_quote gmail_msg" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">On 03/21/2017 04:37 PM, William Burns wrote:<br class="gmail_msg">
> Some users have expressed the need to have some sort of forEach<br class="gmail_msg">
> operation that is performed where the Consumer is called while holding<br class="gmail_msg">
> the lock for the given key and subsequently released after the<br class="gmail_msg">
> Consumer operation completes.<br class="gmail_msg">
<br class="gmail_msg">
Seconding Dan's question - is that intended to be able to modify the<br class="gmail_msg">
entry? In my opinion, sending a function that will work on the<br class="gmail_msg">
ReadWriteEntryView directly to the node is the only reasonable request.<br class="gmail_msg">
I wouldn't like to see blocking operations in there.<br class="gmail_msg"></blockquote><div class="gmail_msg"><br class="gmail_msg"></div></div></div><div dir="ltr" class="gmail_msg"><div class="gmail_quote gmail_msg"><div class="gmail_msg">Hrmm the user can use the FunctionalMap interface for this then it seems? I wonder if this should just be the going in API. I will need to discuss with Galder the semantics of the evalAll/evalMany methods.</div></div></div></blockquote><div><br></div><div>Actually looking at evalAll it seems it doesn't scale as it keeps all entries in memory at once, so this is only for caches with a limited amount of entries.</div><div> </div><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex"><div dir="ltr" class="gmail_msg"><div class="gmail_quote gmail_msg"><div class="gmail_msg"> </div><blockquote class="gmail_quote gmail_msg" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">
<br class="gmail_msg">
><br class="gmail_msg">
> Due to the nature of how streams work with retries and performing the<br class="gmail_msg">
> operation on the primary owner, this works out quite well with forEach<br class="gmail_msg">
> to be done in an efficient way.<br class="gmail_msg">
><br class="gmail_msg">
> The problem is that this only really works well with non tx and<br class="gmail_msg">
> pessimistic tx. This obviously leaves out optimistic tx, which at<br class="gmail_msg">
> first I was a little worried about. But after thinking about it more,<br class="gmail_msg">
> this prelocking and optimistic tx don't really fit that well together<br class="gmail_msg">
> anyways. So I am thinking whenever this operation is performed it<br class="gmail_msg">
> would throw an exception not letting the user use this feature in<br class="gmail_msg">
> optimistic transactions.<br class="gmail_msg">
<br class="gmail_msg">
How exactly reading streams interacts with transactions? Does it wrap<br class="gmail_msg">
read entries into context? This would be a scalability issue.<br class="gmail_msg"></blockquote><div class="gmail_msg"><br class="gmail_msg"></div></div></div><div dir="ltr" class="gmail_msg"><div class="gmail_quote gmail_msg"><div class="gmail_msg">It doesn't wrap read entries into the context for that exact reason. It does however use existing entries in the context to override ones in memory/store.</div></div></div><div dir="ltr" class="gmail_msg"><div class="gmail_quote gmail_msg"><div class="gmail_msg"> </div><blockquote class="gmail_quote gmail_msg" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">
<br class="gmail_msg">
I agree that "locking" should not be exposed with optimistic transactions.<br class="gmail_msg"></blockquote><div class="gmail_msg"><br class="gmail_msg"></div></div></div><div dir="ltr" class="gmail_msg"><div class="gmail_quote gmail_msg"><div class="gmail_msg">Yeah I can't find a good way to do this really and it seems to be opposite of what optimistic transactions are.</div></div></div><div dir="ltr" class="gmail_msg"><div class="gmail_quote gmail_msg"><div class="gmail_msg"> </div><blockquote class="gmail_quote gmail_msg" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">
<br class="gmail_msg">
With pessimistic transactions, how do you expect to handle locking<br class="gmail_msg">
order? For regular operations, user is responsible for setting up some<br class="gmail_msg">
locking order in order to not get a deadlock. With pessimistic<br class="gmail_msg">
transaction, it's the cache itself who will order the calls. Also, if<br class="gmail_msg">
you lock anything that is read, you just end up locking everything (or,<br class="gmail_msg">
getting a deadlock). If you don't it's the same as issuing the lock and<br class="gmail_msg">
reading again (to check the locked value) - but you'd do that internally<br class="gmail_msg">
anyway. Therefore, I don't feel well about pessimistic transactions neither.<br class="gmail_msg"></blockquote><div class="gmail_msg"><br class="gmail_msg"></div></div></div><div dir="ltr" class="gmail_msg"><div class="gmail_quote gmail_msg"><div class="gmail_msg">The lock is done per key only for each invocation. There is no ordering as only one is obtained at a time before it goes to the next. If the user then acquires a lock for another key while in the Consumer this could cause a deadlock if the inverse occurs on a different thread/node, but this is on the user. It is the same as it is today really, except we do the read lock for them before invoking their Consumer.</div></div></div><div dir="ltr" class="gmail_msg"><div class="gmail_quote gmail_msg"><div class="gmail_msg"> </div><blockquote class="gmail_quote gmail_msg" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">
<br class="gmail_msg">
><br class="gmail_msg">
> Another question is what does the API for this look like. I was<br class="gmail_msg">
> debating between 3 options myself:<br class="gmail_msg">
><br class="gmail_msg">
> 1. AdvancedCache.forEachWithLock(BiConsumer<Cache, CacheEntry<K, V>><br class="gmail_msg">
> consumer)<br class="gmail_msg">
><br class="gmail_msg">
> This require the least amount of changes, however the user can't<br class="gmail_msg">
> customize certain parameters that CacheStream currently provides<br class="gmail_msg">
> (listed below - big one being filterKeys).<br class="gmail_msg">
><br class="gmail_msg">
> 2. CacheStream.forEachWithLock(BiConsumer<Cache, CacheEntry<K, V>><br class="gmail_msg">
> consumer)<br class="gmail_msg">
><br class="gmail_msg">
> This method would only be allowed to be invoked on the Stream if no<br class="gmail_msg">
> other intermediate operations were invoked, otherwise an exception<br class="gmail_msg">
> would be thrown. This still gives us access to all of the CacheStream<br class="gmail_msg">
> methods that aren't on the Stream interface (ie.<br class="gmail_msg">
> sequentialDistribution, parallelDistribution, parallel, sequential,<br class="gmail_msg">
> filterKeys, filterKeySegments, distributedBatchSize,<br class="gmail_msg">
> disableRehashAware, timeout).<br class="gmail_msg">
<br class="gmail_msg">
For both options, I don't like Cache being passed around. You should<br class="gmail_msg">
modify the CacheEntry (or some kind of view) directly.<br class="gmail_msg"></blockquote><div class="gmail_msg"><br class="gmail_msg"></div></div></div><div dir="ltr" class="gmail_msg"><div class="gmail_quote gmail_msg"><div class="gmail_msg">I don't know for sure if that is sufficient for the user. Sometimes they may modify another Cache given the value in this one for example, which they could access from the CacheManager of that Cache. Maybe Tristan knows more about some use cases.</div></div></div><div dir="ltr" class="gmail_msg"><div class="gmail_quote gmail_msg"><div class="gmail_msg"> </div><blockquote class="gmail_quote gmail_msg" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">
<br class="gmail_msg">
Radim<br class="gmail_msg">
<br class="gmail_msg">
><br class="gmail_msg">
> 3. LockedStream<CacheEntry<K, V>> AdvancedCache.lockedStream()<br class="gmail_msg">
><br class="gmail_msg">
> This requires the most changes, however the API would be the most<br class="gmail_msg">
> explicit. In this case the LockedStream would only have the methods on<br class="gmail_msg">
> it that are able to be invoked as noted above and forEach.<br class="gmail_msg">
><br class="gmail_msg">
> I personally feel that #3 might be the cleanest, but obviously<br class="gmail_msg">
> requires adding more classes. Let me know what you guys think and if<br class="gmail_msg">
> you think the optimistic exclusion is acceptable.<br class="gmail_msg">
><br class="gmail_msg">
> Thanks,<br class="gmail_msg">
><br class="gmail_msg">
> - Will<br class="gmail_msg">
><br class="gmail_msg">
><br class="gmail_msg">
> _______________________________________________<br class="gmail_msg">
> infinispan-dev mailing list<br class="gmail_msg">
> <a href="mailto:infinispan-dev@lists.jboss.org" class="gmail_msg" target="_blank">infinispan-dev@lists.jboss.org</a><br class="gmail_msg">
> <a href="https://lists.jboss.org/mailman/listinfo/infinispan-dev" rel="noreferrer" class="gmail_msg" target="_blank">https://lists.jboss.org/mailman/listinfo/infinispan-dev</a><br class="gmail_msg">
<br class="gmail_msg">
<br class="gmail_msg">
--<br class="gmail_msg">
Radim Vansa <<a href="mailto:rvansa@redhat.com" class="gmail_msg" target="_blank">rvansa@redhat.com</a>><br class="gmail_msg">
JBoss Performance Team<br class="gmail_msg">
<br class="gmail_msg">
_______________________________________________<br class="gmail_msg">
infinispan-dev mailing list<br class="gmail_msg">
<a href="mailto:infinispan-dev@lists.jboss.org" class="gmail_msg" target="_blank">infinispan-dev@lists.jboss.org</a><br class="gmail_msg">
<a href="https://lists.jboss.org/mailman/listinfo/infinispan-dev" rel="noreferrer" class="gmail_msg" target="_blank">https://lists.jboss.org/mailman/listinfo/infinispan-dev</a><br class="gmail_msg">
</blockquote></div></div></blockquote></div></div>