Writing my network protocol in Rust
After spending a lot of time writing my network protocol in C, I decided that it would be a nice exercise to do the same in Rust. I keep getting back to this language because I want to like it. I hope that having a straightforward task and the passage of time will make things easier than before.
I gotta say, the compiler feels a lot nicer now. Check this out:
I really like it. This feels like the compiler is actively trying to help me get in the right path…
And this error message made me even happier:
Thank you, this is clear, concise and to the point.
I can indeed report that the borrow checker is indeed very much present, but at least now it will tell you what is the magic incantation you need to make it happen:
This is my third or forth foray into Rust, and I stopped a few times before because the learning curve and the… Getting Things Done curve were just weren’t there for me. In this case, however, I’m getting a very different feel all around. There is a lot less messing around and a lot more actually getting to where I want to go. I’m still significantly slower, naturally, since I’m learning idioms and the mechanics of the language, but there is a lot less friction. And the documentation has been top notch, and I’m leaning on that a lot.
In particular, one of my pet peeves seems to have been resolved. Composite error handling is now as simple as:
Removing the ceremony from error handling is a great relief, I have to say.
It took me a couple of evenings to get to the point where I can setup a TCP server, accept a connection and parse the command. Let me see if I can break it up to manageable parts, even though the whole thing is about 100 lines of code, this is packed.
I’m certainly feeling the fact that Rust actually have a runtime library. I mean, C has one, but it is pretty anemic to say the least. And actually having an OOTB solution for packaging is not really an option today, it is a baseline requirement.
Here is the code that handles the connection itself.
By itself, it is pretty simple. It starts by letting the client know that we are ready (the OK msg) and then read a message from the client, parse it and dispatch it. Rinse, repeat, etc.
There are a couple of interesting things going on here that might be worth your attention:
- The read_full_message() function works on bytes, and it is responsible for finding the message boundaries. The code is meant to handle pipelined messages, partial reads, etc.
- Once we have the range of bytes that represent a single message, we translate them to UTF8 string and use string processing to parse the command. This is simpler than the tokenization we did in C.
- After processing a message, we drain the data we already process and continue with the data already in the buffer. Note that we basically reuse the same buffer, so hopefully there shouldn’t be too many allocations along the way.
The buffer handling is the responsibility of the read_full_message() function, which looks like:
I’m using an inefficient method, where I’m reading only 256 bytes at a time from the network, mostly to prove that I can process things that come over in multiple calls. But the basic structure is simple:
- Use TwoWaySearcher to search for a byte pattern in the buffer. This is basically memmem() call.
- If the byte pattern (\r\n\r\n) is found, we have a message boundary and can return that to the caller.
- If the message boundary isn’t found, we need to read more data from the network.
- I’m playing some tricks with the to_scan variable, avoiding the case where I need to scan over data that I have already scanned.
- I’m also validating that I’m never reading too much from the network and abort the connection if we can’t find the message boundary in a reasonable size (8KB).
What remains is some string parsing, which ended up being really easy, since Rust has normal string processing routines.
And this is pretty much it, I gotta say. For that amount of code, it took a long time to get there, but I’m pretty happy with the state of the code. Of course, this is still pretty early in the game and it isn’t really doing anything really interesting. The TCP server can accept only a single connection (breaking the connection will kill the server, at this point), error handling is the same as not having a single catch in the entire system, etc.
What I expect to be… interesting is the use of SSL and concurrent (hopefully async) I/O. We’ll see where that will take us…
Comments
I like the format and structure of your post. I'm sleepy and only dabbled some time ago in rust and still was able to understand everything without a problem.
Interesting post, just one thing: it's "borrow checker", not "burrow checker" ;-)
njy, Thanks, fixed
Comment preview