When creating HttpGetFuture, we don’t actually do anything related to the GET request, which means that the call to Http::get returns immediately with just a simple data structure.
In contrast to earlier examples, we pass in the IP address for localhost instead of the DNS name. We take the same shortcut as before and let connect be blocking and everything else be non-blocking.
The next step is to write the GET request to the server. This will be non-blocking, and we don’t have to wait for it to finish since we’ll be waiting for the response anyway.
The last part of this file is the most important one—the implementation of the Future trait we defined:
ch07/a-coroutine/src/http.rs
impl Future for HttpGetFuture {
type Output = String;
fn poll(&mut self) -> PollState<Self::Output> {
if self.stream.is_none() {
println!(“FIRST POLL – START OPERATION”);
self.write_request();
return PollState::NotReady;
}
let mut buff = vec![0u8; 4096];
loop {
match self.stream.as_mut().unwrap().read(&mut buff) {
Ok(0) => {
let s = String::from_utf8_lossy(&self.buffer);
break PollState::Ready(s.to_string());
}
Ok(n) => {
self.buffer.extend(&buff[0..n]);
continue;
}
Err(e) if e.kind() == ErrorKind::WouldBlock => {
break PollState::NotReady;
}
Err(e) if e.kind() == ErrorKind::Interrupted => {
continue;
}
Err(e) => panic!(“{e:?}”),
}
}
}
}
Okay, so this is where everything happens. The first thing we do is set the associated type called Output to String.
The next thing we do is to check whether this is the first time poll was called or not. We do this by checking if self.stream is None.
If it’s the first time we call poll, we print a message (just so we can see the first time this future was polled), and then we write the GET request to the server.
On the first poll, we return PollState::NotReady, so HttpGetFuture will have to be polled at least once more to actually return any results.
The next part of the function is trying to read data from our TcpStream.
We’ve covered this before, so I’ll make this brief, but there are basically five things that can happen:
- The call successfully returns with 0 bytes read. We’ve read all the data from the stream and have received the entire GET response. We create a String from the data we’ve read and wrap it in PollState::Ready before we return.
- The call successfully returns with n > 0 bytes read. If that’s the case, we read the data into our buffer, append the data into self.buffer, and immediately try to read more data from the stream.
- We get an error of kind WouldBlock. If that’s the case, we know that since we set the stream to non-blocking, the data isn’t ready yet or there is more data but we haven’t received it yet. In that case, we return PollState::NotReady to communicate that more calls to the poll are needed to finish the operation.
- We get an error of kind Interrupted. This is a bit of a special case since reads can be interrupted by a signal. If it does, the usual way to handle the error is to simply try reading once more.
- We get an error that we can’t handle, and since our example does no error handling, we simply panic!
There is one subtle thing I want to point out. We can view this as a very simple state machine with three states:
- Not started, indicated by self.stream being None
- Pending, indicated by self.stream being Some and a read to stream.read returning WouldBlock
- Resolved, indicated by self.stream being Some and a call to stream.read returning 0 bytes
As you see, this model maps nicely to the states reported by the OS when trying to read our TcpStream.
Most leaf futures such as this will be quite simple, and although we didn’t make the states explicit here, it still fits in the state machine model that we’re basing our coroutines around.