The coroutine/wait syntax will have clear similarities to the async/await syntax, although it’s a lot more limited.
The basic rules are as follows:
- Every function prefixed with coroutine will be rewritten to a state machine like the one we wrote.
- The return type of functions marked with coroutine will be rewritten so they return -> impl Future<Output = String> (yes, our syntax will only deal with futures that output a String).
- Only objects implementing the Future trait can be postfixed with .wait. These points will be represented as separate stages in our state machine.
- Functions prefixed with coroutine can call normal functions, but normal functions can’t call coroutine functions and expect anything to happen unless they call poll on them repeatedly until they return PollState::Ready.
Our implementation will make sure that if we write the following code, it will compile to the same state machine we wrote at the start of this chapter(with the exception that all coroutines will return a String):
coroutine fn async_main() {
println!(“Program starting”)
let txt = Http::get(“/1000/HelloWorld”).wait;
println!(“{txt}”);
let txt2 = Http::(“500/HelloWorld2”).wait;
println!(“{txt2}”);
}
But wait. coroutine/wait aren’t valid keywords in Rust. I would get a compilation error if I wrote that!
You’re right. So, I created a small program called corofy that rewrites the coroutine/wait functions into these state machines for us. Let’s explain that quickly.
corofy—the coroutine preprocessor
The best way of rewriting code in Rust is using the macro system. The downside is that it’s not clear exactly what it compiles down to, and expanding the macros is not optimal for our use case since one of the main goals is to take a look at the differences between the code we write and what it transforms into. In addition to that, macros can get quite complex to read and understand unless you work a lot with them on a regular basis.
Instead, corofy is a normal Rust program you can find in the repository under ch07/corofy.
If you enter that folder, you can install the tool globally by writing the following:
cargo install –path .
Now you can use the tool from anywhere. It works by providing it with an input file containing the coroutine/wait syntax, such as corofy ./src/main.rs [optional output file]. If you don’t specify an output file, it will create a file in the same folder postfixed with _corofied.
Note
The tool is extremely limited. The honest reason why is that I want to finish this example before we reach the year 2300, and I finished rewriting the entire Rust compiler from scratch just to give a robust experience using the coroutine/wait keywords.
It turns out that writing transformations like this without access to Rust’s type system is very difficult. The main use case for this tool will be to transform the examples we write here, but it would probably work for slight variations of the same examples as well (like adding more wait points or doing more interesting tasks in between each wait point). Take a look at the README for corofy for more information about its limitations.
One more thing: I assume that you specified no explicit output file going forward so the output file will have the same name as the input file postfixed with _corofied.
The program reads the file you give it and searches for usages of the coroutine keyword. It takes these functions, comments them out (so they’re still in the file), puts them last in the file, and writes out the state machine implementation directly below, indicating what parts of the state machine are the code you actually wrote between the wait points.
Now that I’ve introduced our new tool, it’s time to put it to use.