-
Notifications
You must be signed in to change notification settings - Fork 592
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Introduce async consumer dispatcher #307
Introduce async consumer dispatcher #307
Conversation
Marked it as WIP since I would like to get some feedback first and then take it to finish. Thoughts? // cc @bording @adamralph |
Btw. in my local machine I exclude the WinRT projects for now until I get 👍 or 👎 |
5dc48ee
to
3ce5e10
Compare
The current ConsumerDispatcher has an issue of closure allocation. Because this PR introduces a dedicated path that is internal we can no optimise that path to reduce the closure allocation. I added a commit that removes the Before
After
Will update the description |
Switched to generic method dispatch where possible. It is slightly faster than casting in the async consumer dispatcher BenchmarkDotNet=v0.10.1, OS=Microsoft Windows NT 6.2.9200.0
Processor=Intel(R) Core(TM) i7-4980HQ CPU 2.80GHz, ProcessorCount=8
Frequency=2728066 Hz, Resolution=366.5600 ns, Timer=TSC
[Host] : Clr 4.0.30319.42000, 32bit LegacyJIT-v4.6.1586.0
Job-FPAYGT : Clr 4.0.30319.42000, 64bit RyuJIT-v4.6.1586.0
Job-VVRXBX : Clr 4.0.30319.42000, 32bit LegacyJIT-v4.6.1586.0
|
Implementing an internal interface is a non-breaking change from an API perspective
…now inherits from Connection AutorecoveringConnection and ProtocolBase create correct connection depending on settings
Thank you! I will take a proper look on Monday. Bear in mind the next release will most likely be a major version bump so some breaking changes could be accommodated. |
@kjnilsson Ok then I can remove a few trickeries here. |
@danielmarbach thank you! Do you have any numbers that compare this version with what's currently in master? Also, would be really great to get some feedback from @bording :) |
This is far from done. So no numbers yet. Need to fix the auto recovery connection first
… Am 05.02.2017 um 07:16 schrieb Michael Klishin ***@***.***>:
@danielmarbach thank you! Do you have any numbers that compare this version with what's currently in master?
Also, would be really great to get some feedback from @bording :)
—
You are receiving this because you were mentioned.
Reply to this email directly, view it on GitHub, or mute the thread.
|
@danielmarbach I've taken a bit of a look at your PR today and the overall approach of adding async consumers as an additional, optional feature looks good to me and in terms of that you've got a 👍 . You've outlined some of the benefits above. Once it is a bit more finished it would be nice to try to gather some numbers on any scalability benefits this would provide. General feedback:
Cheers |
I'm in the process of gathering stats. Give me a bit time and I'll update this issue
… Am 06.02.2017 um 18:20 schrieb Karl Nilsson ***@***.***>:
@danielmarbach I've taken a bit of a look at your PR today and the overall approach of adding async consumers as an additional, optional feature looks good to me and in terms of that you've got a 👍 .
You've outlined some of the benefits above. Once it is a bit more finished it would be nice to try to gather some numbers on any scalability benefits this would provide.
General feedback:
I think it is fine to replace AsyncEventHandlers with Func<Args, Task>. Never felt comfortable with mixing multicast delegates with manual acks.
Not sure about the new flag on the ConnectionFactory to enable the special async mode. I'm wondering if there is a more intuitive api we can provide. An AsyncConnectionFactory naturally wouldn't be a completely accurate given only consumption being async. Will think about it a bit more.
Cheers
Karl
—
You are receiving this because you were mentioned.
Reply to this email directly, view it on GitHub, or mute the thread.
|
@michaelklishin Just wanted to say that I've been looking at this with @danielmarbach and so far I like the direction it's heading in. I do think some additional changes are going to be needed, because there's some weird performance issues that we've not been able to sort out yet. |
@bording sounds promising, thank you both! |
I hesitated to do a benchmark with Benchmark.NET because it would burn too many of my CPU cycles ;). Here is a simple test https://gist.github.com/danielmarbach/80b2c60cced407e1d3500db8c85de543 Big caveat: Depending on the clock resolution (on my machine something around 15 miliseconds) the Task.Delay version can be slower. @bording and I ended up playing around with it and pretty much saw good or better results with the async version as longs as Thread.Sleep vs. Task.Delay was a factor of 15 milliseconds. So Thread.Sleep(15) would be Task.Delay(1). If you set Thread.Sleep(1) and Task.Delay(1) then the await version will be 15 times slower because of the internal timer resolution that is used with Task.Delay. So that would be an unfair comparison. Set Thread.Sleep and Task.Delay to a factor of 15 milliseconds. Try to remember that if you play with the results. Here is a local run on my machine When it comes down to the decision whether you should take this PR in or not it is difficult to give a good general advice. You can argue that from an async programming perspective the newer API provides a better async model and in combination with multiple consumers more efficient resource usage. On the other hand it is also a redundant code path which comes with maintainability costs. I'm leaving this up to you. |
@danielmarbach thank you. We would definitely consider it. @kjnilsson is on vacation at the moment, so this can take about a week. By the way, I don't see any benchmark results in you comment. Should there be any? |
Thanks! I will take a proper look on my return. I wouldn't want to compare
an async implementation to a sync on in terms of speed only. It may be more
interesting to instead run tests with very large numbers of consumers and
see how they compare.
Generally I think async consumers is a nice option that some users may
appreciate irrespectively of minor performance differences. @danielmarbach
@bording would you look to update the NSB RabbitMQ binding to use async
consumers?
Cheers
Karl
On Mon, 13 Feb 2017 at 23:59, Michael Klishin <notifications@github.com> wrote:
@danielmarbach <https://github.com/danielmarbach> thank you. We would
definitely consider it. @kjnilsson <https://github.com/kjnilsson> is on
vacation at the moment, so this can take about a week.
By the way, I don't see any benchmark results in you comment. Should there
be any?
—
You are receiving this because you were mentioned.
Reply to this email directly, view it on GitHub
<#307 (comment)>,
or mute the thread
<https://github.com/notifications/unsubscribe-auth/ABIDlMRyNy3_MKBzTiFA3dnqEpY9Co52ks5rcOA9gaJpZM4L26GT>
.
|
@kjnilsson that is exactly the reasoning why I didn't provide any comparison benchmarks. When you compare a single synchronous consumer with a single asynchronous consumer then essentially we are comparing apple with oranges. It is very likely that a single blocking consumer for small operations is faster than the asynchronous one. It is also likely that a single asynchronous consumer is always a bit slower than the sync one. When it starts shining is in terms of resource utilization, lower CPU usage and better API usage with asynchronous implementors in multi-consumer scenarios. |
@kjnilsson That was certainly the intent behind investigating this PR. We do already have a pretty viable approach with how things are without this, so I'd definitely want to do some additional perf testing to see if ended up making sense to move over to this. |
@bording ok thanks, I took a look at your current code for this. It will be interesting to see how it compares. :) |
Should we consider this for merging or there's more to come? |
From my side not unless you have specific change requests. Thought Karl wants to review it first
… Am 24.02.2017 um 18:39 schrieb Michael Klishin ***@***.***>:
Should we consider this for merging or there's more to come?
—
You are receiving this because you were mentioned.
Reply to this email directly, view it on GitHub, or mute the thread.
|
Apologies @danielmarbach , I was holding back a bit as I got the impression you and @bording were looking at a performance issue. I will spend some time on this PR this week. We would need some tests for this before merging. I will probably add some as part of the review. |
@kjnilsson The performance issue I mentioned here ended up being related to the Task.Delay stuff that @danielmarbach mentioned here, so it turned out to not actually be an issue once we had our tests comparing the same results. Sorry for the confusion if that wasn't clear enough! I do still want to compare the usage of the async consumer to the original in our NSB RabbitMQ transport, but I haven't gotten to that yet. I wouldn't expect that decision to hold this PR up, since there is value in having the async consumer regardless. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This looks good to me and I'm happy to merge this into master and release a preview. I have republished the changes in a separate branch where I am adding some tests. I want to write a couple of more tests then I will merge.
Ok this has been manually merged with a couple of added tests. Closing. Thank you! |
I've published a pre-release with these changes here: https://www.nuget.org/packages/RabbitMQ.Client/5.0.0-pre3 |
Btw @kjnilsson the changes could go into a minor. doesn't have to be a major bump. |
@kjnilsson why has the major version been bumped to 5? |
There are other breaking changes in the current master.
…On Wed, 15 Mar 2017 at 19:58, Adam Ralph ***@***.***> wrote:
@kjnilsson <https://github.com/kjnilsson> why has the major version been
bumped to 5?
—
You are receiving this because you were mentioned.
Reply to this email directly, view it on GitHub
<#307 (comment)>,
or mute the thread
<https://github.com/notifications/unsubscribe-auth/ABIDlFxj_LvhjWiHqc0y7sReeVYAB8tQks5rmELUgaJpZM4L26GT>
.
|
In the WorkPool, why do you need to spin every 100ms? Why cant you use Timeout.Infinite and await cancellation from Stop() and new message from Enqueue(...)? It seems to me that spinning is useless. |
@thtp this PR is a result of at least 2 different approaches that were investigated and a fair amount of benchmarking. Do you have any numbers to back your claim or is it a gut feeling? I'd recommend submitting a pull request over claiming that the approach that was adopted is useless. It would be a lot more convincing than random comments. |
Look at AsyncConsumerWorkService.WorkPool. It does exactly what you propose. Except that it knows it's single worker looping nature and does not use interlocked. The aim of the PR was not to touch the sync path but to introduce a new async code. Is it possible to do better in the sync version? Probably yes. Why don't you give it a try? Optimizations can always be introduced and learning happens over time ;)
|
Before anyone spends a lot of time on the sync dispatcher, keep in mind that we still hope to begin working on a new from scratch .NET client this year and there may or may not be a place for the sync dispatcher there. |
I was following an idea to see if it would be possible to make a tiny step in a minor version to become async on the path where the client dispatches to the consumers. I was able to get it going by doing the following:
IAsyncConnection
Async
flag on the connection factory which allows to opt-in for the async consumer dispatcherIConnection
with anIAsyncConnection
when the connection factory flag is setIAsyncBasicConsumer
for fully asynchronous consumersIAsyncConnection
is used you can only use theIAsyncBasicConsumer
implementationsAsyncConsumerDispatcher
stick toIConsumerDispatcher
but internally enqueuesFunc<Task>
instead ofAction
Task.Run
with an async body. The idle phase is implemented with aTaskCompletionSource
and aTask.WhenAny
that either completes when the delay is due or the TCS is set. Use atomic field reference assigned to create a newTaskCompletionSource
after each iteration (this should be fine since we only have one loop and concurrent enqueues should still work due to TrySetResult)Func<Task>
allocations per call on the consumer dispatcher. So that means we save one allocation per callBefore
After
Benefits
Tradeoffs
AsyncEventHandler
to stick as much as possible to the current way how consumers work although probably aFunc<Args, Task>
would be more elegant.GetAwaiter().GetResult()
calls (shudder)