Site tech stack 2: the unfathomed depths
Updated 2024-03-27 / Created 2024-03-27 / 1.87k words
RSAPI and the rest of my infrastructure.Transhumanism is attractive until you have seen how software is built.
The original Site tech stack article (updated since release somewhat as hardware has improved and software been replaced) covers the basic workings of the public-facing website. However, I run other things, some of which are interesting to talk about! I have a number of services for personal use running on the same infrastructure, and several non-web-facing but public services. Here's the latest edition of the handy diagram I made in Graphviz:
Expand
This is split into several boxes indicating the various servers several subsystems run on. As I mention in the comments of the old article, I have a physical server running the actual compute tasks (protagonism
), but a VPS (procyon
) is where your HTTP requests are initially going. Since it tends to have better uptime[1], it also runs the uptime monitoring system onstat3, my Discord bot, and a few other directly network-facing things: osmarksDNS[2], an IRC server (APIONET) and smtp2rss.
If you know what SMTP stands for, hearing "smtp2rss" may have confused you (if you have not talked to me much) or worried you (if you have). Don't worry: it's perfectly sane and reasonable. I like RSS, but many people try to email me things, without an RSS fallback[3]. I dislike minor workflow inconveniences and am willing to throw arbitrary amounts of technology and engineering at them (sometimes), so I wrote a Python script to take inbound emails on a spare domain and expose them as RSS feeds. As a handy bonus, it can provide disposable mailboxes for signing up to services.
A fun quirk of the nginx installation on procyon
is that, since I wanted it to not be able to decrypt requests to protagonism
(I don't entirely trust it, and duplicating the certificate issuance programs on each would be irritating), I use ngx_stream_ssl_preread to forward still-encrypted TLS connections either to itself (on another port) or protagonism
's reverse proxy. As janky as this sounds, it does seem to work fine, except for one extremely-hard-to-reproduce bug I suspect might be related where users sometimes get shown 404 pages or the status page incorrectly. Traffic is routed over Tailscale using Headscale[4].
Several of protagonism
's services are mostly-self-contained personal-use applications, such as Minoteaur (notes), ankisyncd (flashcards), atuin (shell history) and calibre-web (books). The rest are somewhat more interesting, in that they do more and are in some cases publicly accessible. For example, SPUDNET. It was built to serve the needs of an "operating system" for ComputerCraft (a Minecraft computer mod) by providing remote debugging services. Originally built about six years ago, it somehow still works with relatively minor changes (new protocol support). It provides bidirectional many-to-one and many-to-many communications over websocket, with an unnecessarily sophisticated authentication system, as well as HTTP long polling fallbacks and incident reports. Skynet is a somewhat simpler version.
I also have a monitoring system using VictoriaMetrics[5] and Grafana. VictoriaMetrics periodically scrapes services for metrics and stores time series, and Grafana can plot them. This is a fairly standard setup, and lots of software exposes Prometheus-compatible metrics itself or has an exporter available (e.g. the node_exporter for general Linux machine status and the PostgreSQL exporter). I went slightly further by exposing metrics in most of my custom applications, so I have, for instance, nice dashboards from my Discord bot. These used to be public, but apparently that exposing any dashboard in Grafana allows users to read any data out of the backend, which was a bit of a security issue. One might reasonably question how much use I get out of these, as I don't get enough traffic to have to debug performance issues much, but they do look nice and their presence is calming.
The most important component I have is undoubtedly RSAPI, the highly custom integration script which does about twenty different things since putting them in different services would have been annoying. Many things I need to do share the same basic building blocks - a database or simple state storage, timers and an HTTP server - and while it would be possible though tricky to factor this out into a library and write several microservices, the deployment would be harder to manage with my tools and many of the parts have to interoperate anyway. Some would consider a 1600-line monolithic Python program plugged into 10 different APIs "bad", but, freed from Conway's law[6], I think it is actually the most efficient way to do this. For whatever reason I can track its structure mentally very efficiently, so it's not hard to work on.
RSAPI has a wide range of functions, having grown from a short Flask application which served fortunes to PotatOS by accretion of additional capabilities as they were needed. The exact history has been lost to the halcyon days of poor version control and backups, but it was built in roughly this order:
- Initial version built: served fortunes and Seventy Maxims of Maximally Effective Mercenaries over HTTP.
- "Currently playing" support for my internet radio server, via integration with MPD.
- youtube-dl web frontend and very basic login.
- IRC bot for server status and MPD status.
- Rewrite from Flask/gevent to asyncio/aiohttp.
- DNS to comments (and IRC) bridge - converts specially formatted DNS queries to a subdomain to comments on a certain page.
- "Currently playing" support expanded with listener counts.
- Miniflux (RSS reader) to Discord bridge, and (unnecessary, since it has an RSS button I somehow ignored) RoyalRoad fiction to RSS bridge.
- Random video selector (from local media folders).
- IncDec/IRC Bridge.
- Cancelled rewrite to make it more modular. Migration of internal databases from SQLite to LevelDB.
- Live chat on internet radio (specialized bridge to IRC).
- NASA Astronomy Picture of the Day fetcher implemented. Daily task scheduler built.
- Wordnik Word of the Day fetcher implemented.
- ComputerCraft to Prometheus metrics bridge.
- "Webmaze" prototype - an infinite partly connected 3D grid implemented fully statelessly, intended for a hypothetical adventure game which never materialized.
- Freefall (webcomic) fetcher implemented.
- Migration from youtube-dl to yt_dlp.
- Cross-device scratchpad/clipboard for copying short text between computers.
- For some reason, a script which infinitely generates primes, digits of e and digits of pi, reads them out using bad TTS, and sends them as an internet radio stream[7].
- Random bytes API.
- Basic service status page for a tablet (to be glued to walls).
- Migration back to SQLite. Proper database migration system.
- URL shortener - in conjunction with my really short domain, it can produce really short URLs, as well as ZWS-based URLs and two-word URLs like https://0t.lt/YearningPried - these use a special prefix-free wordlist so they can also be typed in uncapitalized unambiguously.
- zzcxz telephony interface.
- Tablet status interface enhanced with list of failed services (from systemd).
- Personal event logger.
- Remote to local calendar synchronizer.
- Weather and calendar updates (for the tablet).
- Threat Update (Twitter) scraper.
- SwitchCraft player surveillance (Dynmap).
- RSS feed for random memes (for XScreensaver).
- Better login system - multiple users, SSO for other services via Nginx authentication subrequests, basicauth option for non-interactive systems.
- Key/value storage backend for PotatOS, due to the shutdown of the random free API it used before.
- Internal LLM-based Threat Updates system[8], to replace the archive of historical ones and Twitter scraper. I was too lazy to work out how to draw nicely line-wrapped text in images in Python, so this actually invokes a ComputerCraft emulator, runs the Threat Update implementation on that, dumps its virtual screen, and renders that to an image.
- The new comments system, replacing Isso. It supports ominous AI faces (from StyleGAN2, thanks to StyleGANCpp[9]), leftvotes/rightvotes for greater user expression, SSO integration, better threading, and lower client resource use.
- Lighting control orchestration.
I'm especially proud of the ComputerCraft to Prometheus metrics bridge. While it's only about 20 lines of code (plus the ComputerCraft side), it does allow me to feel cool about being able to monitor meaninglessly large numbers in great detail. There are similar Factorio mods, which I'll probably use next time I play.
I also have some custom inference servers backing Meme Search Engine and Maghammer, in addition to an ExllamaV2-based LLM API used in PotatOS. Early prototypes loaded the models in-process, but this was very inflexible: restarts were slow, only one process at a time could use them, and it effectively required that consuming code be written in Python. The servers are basic (no automatic batching and few optimizations), but are presently good enough to handle traffic. The CLIP one is in fact open as part of Meme Search Engine.
I haven't covered every osmarks.net service in this post, or even all the ones in the slightly outdated diagram above, but I think I got the most interesting ones. I hope this was informative, and did not accidentally make people notice horrible security issues I missed.