Now that we have an overview of our design and know what to do, we can go on and make the necessary changes to our program, so let’s go through each file we need to change. We’ll start with main.rs.
main.rs
We already made some changes to main.rs when we ran corofy on our updated coroutine/wait example. I’ll just point out the change here so that you don’t miss it since there is really nothing more we need to change here.
Instead of polling the future in the main function, we created a new Runtime struct and passed the future as an argument to the Runtime::block_on method. There are no more changes that we need to in this file. Our main function changed to this:
ch08/a-runtime/src/main.rs
fn main() {
let future = async_main();
let mut runtime = Runtime::new();
runtime.block_on(future);
}
The logic we had in the main function has now moved into the runtime module, and that’s also where we need to change the code that polls the future to completion from what we had earlier.
The next step will, therefore, be to open runtime.rs.
runtime.rs
The first thing we do in runtime.rs is pull in the dependencies we need:
ch08/a-runtime/src/runtime.rs
use crate::future::{Future, PollState};
use mio::{Events, Poll, Registry};
use std::sync::OnceLock;
The next step is to create a static variable called REGISTRY. If you remember, Registry is the way we register interest in events with our Poll instance. We want to register interest in events on our TcpStream when making the actual HTTP GET request. We could have Http::get accept a Registry struct that it stored for later use, but we want to keep the API clean, and instead, we want to access Registry inside HttpGetFuture without having to pass it around as a reference:
ch08/a-runtime/src/runtime.rs
static REGISTRY: OnceLock<Registry> = OnceLock::new();
pub fn registry() -> &’static Registry {
REGISTRY.get().expect(“Called outside a runtime context”)
}
We use std::sync::OnceLock so that we can initialize REGISTRY when the runtime starts, thereby preventing anyone (including ourselves) from calling Http::get without having a Runtime instance running. If we did call Http::get without having our runtime initialized, it would panic since the only public way to access it outside the runtime module is through the pub fn registry(){…} function, and that call would fail.
Note
We might as well have used a thread-local static variable using the thread_local! macro from the standard library, but we’ll need to access this from multiple threads when we expand the example later in this chapter, so we start the design with this in mind.
The next thing we add is a Runtime struct:
ch08/a-runtime/src/runtime.rs
pub struct Runtime {
poll: Poll,
}
For now, our runtime will only store a Poll instance. The interesting part is in the implementation of Runtime. Since it’s not too long, I’ll present the whole implementation here and explain it next:
ch08/a-runtime/src/runtime.rs
impl Runtime {
pub fn new() -> Self {
let poll = Poll::new().unwrap();
let registry = poll.registry().try_clone().unwrap();
REGISTRY.set(registry).unwrap();
Self { poll }
}
pub fn block_on<F>(&mut self, future: F)
where
F: Future<Output = String>,
{
let mut future = future;
loop {
match future.poll() {
PollState::NotReady => {
println!(“Schedule other tasks\n”);
let mut events = Events::with_capacity(100);
self.poll.poll(&mut events, None).unwrap();
}
PollState::Ready(_) => break,
}
}
}
}
The first thing we do is create a new function. This will initialize our runtime and set everything we need up. We create a new Poll instance, and from the Poll instance, we get an owned version of Registry. If you remember from Chapter 4, this is one of the methods we mentioned but didn’t implement in our example. However, here, we take advantage of the ability to split the two pieces up.
We store Registry in the REGISTRY global variable so that we can access it from the http module later on without having a reference to the runtime itself.