Unexpected complexity writing an async Spartan protocol server

=> home

Spartan[1] is a cool smol internet protocol. It's even simpler than Gemini. There's no TLS, no need for URL parsing and even less status codes! I just wrote my own Spartan client and server - Spartoi[2] in an afternoon, for fun. What I'm not expecting is some parts of it being (slightly) more complicated than Gemini. Maybe my server deisgn was not what the spartan creator expected.

Also, I'm not saying Spartan is a bad protocol. It does it's job of being really simple. Just I'm not expecting the server being more complicated than Gemini.

=> [1]: The Spartan Protocol Homepage | [2]: Spartoi - Spartan server/client for the Drogon web application framework

Spartan requests are composed of a host, path, content size and content itself. A trivial request would look like the following. Like HTTP (not HTTPS) the host is directly transmitted in the TCP stream. The path is a string of characters. Then the content size is a number. 0 means no content.

example.com /file.gmi 0

And a request with content would look like this:

example.com /forum/post_new 12
Hello World!

This enables Spartan to support uploading files. (Unlike Gemini, which files has to be base64 encoded and squeezed into the URL. Which has a 1024 byte limit.)

example.com /upload/ubuntu.iso 

Like my Gemini library, dremini[3]. Spartoi is based on trantor[4]. Which forces everything to be written in an asynchronous fashion. In Gemini, the entire request is just a URL followed by CRLF. I could just scan for CRLF in the TCP stream receive callback. And the entire parser would be stateless.

=> [3] Dremini - Gemini server/client for the Drogon web application framework | [4]: trantor - a non-blocking I/O tcp network lib based on c++14/17

server_.setRecvMessageCallback([&](const TcpConnectionPtr& conn, MsgBuffer* buf) {
    auto crlf = buf->findCRLF();
    if(crlf != nullptr)
        process_request(conn, buf->peek(), crlf);
});

Legit that's it. But with Spartan. I need to store the expected body size and and a half-generated request in the per-connection context. And handle the edgecase of everything being sent in one TCP packet.

server_.setRecvMessageCallback([&](const TcpConnectionPtr& conn, MsgBuffer* buf) {
    auto state = conn->getContext();
    if(state.request != nullptr &&
        buf->readableBytes() >= state.request->content_size) {
        ... // Handle the request
        process_request(conn, req);
    }


    auto crlf = buf->findCRLF();
    buf->retrieve(std::distance(buf->peek(), crlf));
    // since there's a chace we see a small MTU, thus the request
    //  line may be split into multiple packets
    if(crlf == nullptr)
        return;

    auto [req, content_length] = parseSpartanRequest(buf->peek(), crlf);
    ...

    conn->setContext(SpartanState{req, content_length});

    // Everything is transmitted in a single packet, we won't receive any
    // more data. We are forced to handle the request now.
    if(buf->readableBytes() >= content_length)
        process_request(conn, req);
});

Spartan's design makes perfect sense in a blocking environment. But when your framework forces you to be asynchronous. Spartan can get more complicated. Funny a smol net protocol reminded me how writing asynchronous code can make your problem at hand get exponentially harder.

Proxy Information
Original URL
gemini://gemini.clehaxze.tw/gemlog/2022/06-11-unexpected-complexity-writing-an-async-spartan-protocol-server.gmi
Status Code
Success (20)
Meta
text/gemini
Capsule Response Time
1793.914512 milliseconds
Gemini-to-HTML Time
0.469407 milliseconds

This content has been proxied by September (ba2dc).