Richard's April Update
Apr 30, 2020Boost 1.73 Released and other Matters
The 1.73.0 release of Boost took up more attention than I had anticipated, but in the end all seemed to go well.
Since then I’ve been working through the issues list on GitHub and am now starting to make some headway.
I cam across a few other interesting (to me) topics this month.
(Possibly) Interesting Asio Things
Last month I asked the question, “Is it possible to write an asynchronous composed operation entirely as a lambda?”.
This month I went a little further with two items that interested me.
The first is whether asio’s async_compose
can be adapted so that we can implement a complex composed operation involving
more than one IO object easily using the asio faux coroutine
mechanism.
The second was whether is was possible to easily implement an async future in Asio.
Async Asio Future
Here is my motivating use case:
The salient points here are:
- no matter on which thread the promise is fulfilled, the future will complete on the associated executor of the handler
passed to
async_wait
- Ideally the promise/future should not make use of mutexes un-necessarily.
- (problematic for ASIO) It must work with objects that are not default-constructable.
In the end, I didn’t achieve the second goal as this was not a priority project, but I would be interested to see if anyone can improve on the design.
The source code is here
I tried a couple of ways around the non-default-constructable requirement. My first was to require the CompletionToken to the async_wait initiating function to be compatible with:
But I felt this was unwieldy.
Then I remembered Boost.Outcome. I have been looking for a use for this library for some time.
It turns out that you can legally write an ASIO composed operation who’s handler takes a single
argument of any type, and this will translate cleanly when used with net::use_future
, net::use_awaitable
etc.
A default Boost.Outcome object almost fits the bill, except that its exception_ptr type is boost rather than standard.
This is easily solved with a typedef:
I was feeling please with myself for figuring this out, until I came to test code code under C++11… and realised that Boost.Outcome is only compatible with C++14 or higher.
So in the end, I cobbled together a ‘good enough’ version of outcome using a variant:
The code for this is here
Finally this allowed me to express intent at the call site like so:
The coroutine interface can be made cleaner:
For the above code to compile we’d have to add the following trivial transform:
Easy Complex Coroutines with async_compose
When your composed operation’s intermediate completion handlers are invoked,
the underlying detail::composed_op
provides a mutable reference to itself. A typical completion handler looks like
this:
What I wanted was a composed operation where the following is legal:
Which I think looks reasonably clear and easy to follow.
In this work I had to overcome two problems - writing the framework to allow it, and thinking of a maintainable way to express intent in the interrelationships between the asynchronous operations on the timer and the socket.
Solving the copyable composed_op problem was easy. I did what I always do in situations like this. I cheated.
asio::async_compose
produces a specialisation of a detail::composed_op<>
template. Substituting a disregard of the
rules for knowledge and skill, I simply reached into the guts of asio and produced a copyable wrapper to this class.
I also cut/pasted some ancillary free functions in order to make asio work nicely with my new class:
Here’s the code… it’s not pretty:
With that in hand, and with a little more jiggery pokery, I was able to express intent thus:
The full code can be seen here
There are a couple of interesting things to note:
If you start two or more async operations that will complete on the same object, they must all be allowed to complete. This is why we yield and wait for both the socket and the timeout:
This leads directly to the problem of managing the error_code. Two error_codes will be produced - one for the timer (which we hope to cancel before it times out) and one for the resolve operation. This means we have to store the first relevant error code somewhere:
And we need a means of allowing communication between the timeout timer and the resolver:
One cancels the other….
In the end I was unsure how much is gained, other than pretty code (which does have value in itself).
Unified WebClient
Exploratory work started on the unified web client. After some discussion, Vinnie and I agreed on the following design decisions:
- Interface to model closely the very popular Python Requests module.
- Sync and Async modes available.
- Homogenous (mostly non-template) interface, behind which system-specific implementations can reside.
- Where native library support is available, that will be used,
- Where not, internally the library will be implemented in Asio/Beast.
- Coroutine friendly.
Once more progress has been made on the Boost.Beast issue tracker, I will be focusing attention here.
All Posts by This Author
- 08/10/2022 Richard's August Update
- 10/10/2021 Richard's October Update
- 05/30/2021 Richard's May 2021 Update
- 04/30/2021 Richard's April Update
- 03/30/2021 Richard's February/March Update
- 01/31/2021 Richard's January Update
- 01/01/2021 Richard's New Year Update - Reusable HTTP Connections
- 12/22/2020 Richard's November/December Update
- 10/31/2020 Richard's October Update
- 09/30/2020 Richard's September Update
- 09/01/2020 Richard's August Update
- 08/01/2020 Richard's July Update
- 07/01/2020 Richard's May/June Update
- 04/30/2020 Richard's April Update
- 03/31/2020 Richard's March Update
- 02/29/2020 Richard's February Update
- 01/31/2020 Richard's January Update
- View All Posts...