TCP Server in Zig - Part 2 - Message Boundaries
Oct 05, 2024
When thinking about communication between systems using TCP, we generally think about it in terms of messages: an HTTP response, a database record, etc. The implementation though comes down to writing and reading bytes. We need a way to serialize our messages to bytes and parse bytes back into our messages. To this end, you might decide to use JSON. For example, to send a User
over TCP, we might try to do something like:
{
var message = try std.json.stringifyAlloc(allocator, user, .{});
defer allocator.free(message);
_ = try posix.write(socket, message);
}
And, to read this, we'd try:
var buf: [4096]u8 = undefined;
const n = try posix.read(socket, &buf)
var parsed = try std.json.parseFromSlice(User, allocator, buf[0..n], .{})
defer parsed.deinit()
const user = parsed.value;
The problem is that this isn't going to work. Actually, the real problem is that something like this might work fairly reliably if you're testing locally, making you think that it's working, but it will definitely not work reliably in a production environment.
It's tempting to think that if the sender does 1 write the receiver has to do 1 read. But TCP is just a continuous stream of bytes, it has no concept of message boundaries. It's possible, and quite common, that a single write will require multiple reads. In fact, if you write a 100 byte message, the receiver might need to do anywhere from 1 to 100 reads. Conversely, if you issue 2 or more writes, the other side might get all that data in a single read.
There's no getting around this, so when we write a message, we need to add our own message boundaries which allows the receiving end to piece the parts together whether it takes 1, 2 or 100 reads.
One way to do this is to delimit each message with one or more special character. This has a number of benefits. First of all, it's simple. Second, a lot of libraries, including Zig's standard library, have facilities for reading from a source until a delimiter is reached. Finally, it's almost perfect for streaming: the sender doesn't need to have the full message ahead of time and can just send data as it becomes available, sending a final delimiter at the end. A major downside is that that delimiter can't be part of the data being sent - making this approach impractical if you're sending binary data. Also, while it might have efficiency advantages for the sender in some cases, it might remove some optimization opportunities from the receiver since the size of the message isn't known until the full message is received.
On the write side, adding a delimiter is easy. For this example, we'll use 0 as our delimiter:
fn writeMessage(socket: posix.socket_t, message: []const u8) !void {
try writeAll(socket, message);
if (try posix.write(socket, &[1]u8{0}) != 1) {
return error.Closed;
}
}
fn writeAll(socket: posix.socket_t, msg: []const u8) !void {
var pos: usize = 0;
while (pos < msg.len) {
const written = try posix.write(socket, msg[pos..]);
if (written == 0) {
return error.Closed;
}
pos += written;
}
}
On the read side, things are a bit more complicated. Here's an implementation that isn't quite right:
fn readMessage(socket: posix.socket_t, buf: []u8) ![]u8 {
var pos: usize = 0;
while (true) {
const n = try posix.read(socket, buf[pos..]);
if (n == 0) {
return error.Closed;
}
const end = pos + n;
const index = std.mem.indexOfScalar(u8, buf[pos..end], 0) orelse {
pos = end;
continue;
};
return buf[0..pos + index];
}
}
There are two issues with this code, can you spot them? The first is that error.Closed
will be returned if the message is larger than our buffer (pos
will become equal to buf.len
and the slice passed into read
will have a length of zero). The second issue is relevant to the discussion of message boundaries: we might read more than a single message. In the next part, we'll see how that can be a good thing, but with the above implementation, the extra data is lost (corrupting the next message).
If you're not quite sure how we might read and then throwaway data from subsequent messages, consider the case where the sender write "hello\0" followed by "world\0". As we've said before, it doesn't matter how many calls to write
it takes to send the data. On the receiving side, using our readMessage
function and a buf
with a length of 128, a single call to read
could populate buf
with any of the following:
h
he
hel
hell
hello
hello\0
hello\0w
hello\0wo
hello\0wor
hello\0worl
hello\0world
hello\0world\0
And, let's say that a read got 2 bytes, "he", the subsequent read might, for example, get "llo\0wor". Unless we get lucky and happen to read "hello\0" in a 1 or more calls to read
, any extra data we read is lost.
Again, this is something that we'll explore more in the next part. Reading more than 1 message at a time is actually something we want, but we'll need to add state to handle these situations. Having said that, std.net.Stream
does expose an std.io.Reader
interface which could be wrapped in a std.io.BufferedReader
to make all of this easier, but besides doing it ourselves for greater control and for learning, Zig's implementation is far from ideal.
Before we can look at the second way to add message boundaries, we need to look at how integers can be encoded as bytes. One option is to encode numbers as text, using a text-encoding format like UTF8. If you run:
const std = @import("std");
pub fn main() !void {
std.debug.print("{any}\n", .{"123"});
}
you'll get: {49, 50, 51}. This is because our Zig source code is encoded in UTF8 and the UTF8-encoded values for 1, 2 and 3 are: 49, 50 and 51. (
While text-encoding integers are human-readable, the conversion of an integer to its text representation and back again isn't the most efficient. As we'll see in the next part though, the real issue is that the length of the resulting text depends on the number of digits: a 1 digit number takes 1 byte, a 2 digit number takes 2 bytes, etc. Instead of text encoding we can use binary encoding, which allows us to encode a large range of integers with a fixed number of bytes. For example, using only 4 bytes (which is 32 bits) we can store integers up to any integer from 0 - 4,294,967,29.
Binary encoding is pretty easy. Using 4 bytes, 0 is {0, 0, 0, 0}, 1 is {0, 0, 0, 1}, 2 is {0, 0, 0, 2}, and so on. Eventually we reach 255 which is {0, 0, 0, 255}. But what's 256 - a byte (a u8
) is limited 256 values (0-255). All we need to do is increment the next byte by 1 and start over so that 256 is {0, 0, 1, 0} and 257 is {0, 0, 1, 1}. Can you tell how the largest number, 4,294,967,295, is stored? It's {255, 255, 255, 255}.
There is a variation of this format where 1 is {1, 0, 0, 0} and 2 is {2, 0, 0, 0} and 256 is {0, 1, 0, 0} and 257 is {1, 1, 0, 0}. The first way that we looked at, with the least significant bits on the right, is called big-endian. The second way, with the least significant bits on the left, is called little-endian. Both of these are commonly used and there might be very slight performance differences between them in some cases. The important thing, when we're talking about communication, is that both sides agree of which encoding is being used.
Chances are that, internally, you computer uses little-endian encoding to store numbers. You can figure it out by running:
const std = @import("std");
pub fn main() !void {
const x : u32 = 257;
std.debug.print("{any}\n", .{std.mem.asBytes(&x)});
}
You'll almost certainly see: { 1, 1, 0, 0 }. Alternatively, you could print out the value of @import("builtin").cpu.arch.endian()
. In Zig, you use the std.mem.writeInt
function to turn an integer into a little or big-endian encoded array:
const std = @import("std");
pub fn main() !void {
var buf: [4]u8 = undefined;
std.mem.writeInt(u32, &buf, 257, .big);
std.debug.print("big-endian: {any}\n", .{&buf});
std.mem.writeInt(u32, &buf, 257, .little);
std.debug.print("little-endian {any}\n", .{&buf});
}
Will give you:
big-endian: { 0, 0, 1, 1 }
little-endian: { 1, 1, 0, 0 }
As we'll see next, having a way to encode integers using a fixed number of bytes allows us to prefix each message with its length, which is an easy and efficient way to add boundaries to messages.
The other way we can add boundaries is to prefix each message with a header. The simplest header would be the length of the message. For example, using what we learnt above, if we wanted to send "hello!" using a 4 byte little-endian encoded length-prefix, we'd send: {6, 0, 0, 0, 'h', 'e', 'l', 'l', 'o', '!'}.
Writing and reading these types of messages is straightforward. For now, we'll stick with an explicit implementation, but in the next part, we'll optimize both the write and the read.
fn writeMessage(socket: posix.socket_t, msg: []const u8) !void {
var buf: [4]u8 = undefined;
std.mem.writeInt(u32, &buf, @intCast(msg.len), .little);
try writeAll(socket, &buf);
try writeAll(socket, msg);
}
fn writeAll(socket: posix.socket_t, msg: []const u8) !void {
var pos: usize = 0;
while (pos < msg.len) {
const written = try posix.write(socket, msg[pos..]);
if (written == 0) {
return error.Closed;
}
pos += written;
}
}
We need to @intCast
the msg.len
from a usize
to the u32
which we've told std.mem.writeInt
to expect. But besides that, when adding a delimiter, we wrote the message and then the delimiter. Here, with a header prefix, we write the header and then the message.
The simplest, but not most efficient, way to read this type of message is to fill a 4-byte buffer with the header prefix, decode the length, and then read exactly len
bytes:
fn readMessage(socket: posix.socket_t, buf: []u8) ![]u8 {
var header: [4]u8 = undefined;
try readAll(socket, &header);
const len = std.mem.readInt(u32, &header, .little);
if (len > buf.len) {
return error.BufferTooSmall;
}
const msg = buf[0..len];
try readAll(socket, msg);
return msg;
}
fn readAll(socket: posix.socket_t, buf: []u8) !void {
var into = buf;
while (into.len > 0) {
const n = try posix.read(socket, into);
if (n == 0) {
return error.Closed;
}
into = into[n..];
}
}
The new readAll
function reads from the socket until the provided buffer is full - no more or less. This means we can read our 4 byte header, decode it and, knowing the length of the message, read exactly the right number of bytes.
If you don't need to support messages larger than 16K (2^16), you can use a 2 byte-length prefix instead. Also, many binary protocols tend to have a header which is more than just the length. You commonly see headers that contain both a length and a type. PostgreSQL's binary protocol is comprised of a 1 byte message type followed by a 4 byte big-endian encoded length followed by the actual message. WebSocket, on the other hand, has a variable length header. An entire WebSocket message can be as short as 2 bytes, but the longest message will have a header that's 14 bytes - it angers me.
You might be surprised to know that some protocols use both delimiters and some type of prefix. HTTP, for example, uses delimiters for its headers, but the body's length is typically defined by the text-encoded Content-Length header. Redis also stands out as having a mix of both delimiters (for ease of human-readability) and text-encoded length prefix.
Personally, if you are making up your own protocol, I'd recommend going with a binary-encoded fixed-length header which includes both a message type and a payload length (like PostgreSQL). It's simple to implement and can be efficient to read and write.
The rest of this series is not going to focus on message boundaries. The point of the series is to look at the mechanics of writing a TCP server - threads and sockets and such. Still, I've often seen newcomers to network programming struggle with this this topic. Hopefully, this overview was worth the distraction from our core focus.