NYT_Games: check it out.. new daily crossword.. the ‘Midi!’

jswanson: welp, there goes another four minutes of my day

NYT_Games: huzzah!

jswanson: s/four/slightly over two/
NYT_Games: check it out.. new daily crossword.. the ‘Midi!’

jswanson: welp, there goes another four minutes of my day

NYT_Games: huzzah!

jswanson: s/four/slightly over two/
I’ve been sitting on a Bluesky Personal Data Server (PDS) for a few months now. Why? Great question.. I guess because a few of my privacy-nerd friends were asking if anyone had tried to set one up. I like setting stuff up so I did, and am finally getting around to documenting the thing. I put together a guide for others who want to do something similar: bluesky-pds-guide.
The AT Protocol (Authenticated Transfer Protocol) was created by the Bluesky team. When you use Twitter or Facebook, your data lives on their servers. You’re locked in. A platform changes policies, gets acquired, shuts down—you lose everything.
The AT Protocol flips this. Instead of one company owning everyone’s data, you can run your own Personal Data Server (PDS). Your posts, follows, and media live on your server. You can move between PDS providers, or run your own, without losing your identity.
It’s basically how email works. You can have Gmail or run your own mail server, but you can still email anyone. The AT Protocol brings that same interoperability to social media. It’s the antidote to the ‘walled garden’ junk we’ve been dealing with for a decade. Bluesky started sidling away from their open stance a bit, my friends got nervous, so here we are.
I set up a PDS at jswan.dev. Running:
| Component | Choice |
|---|---|
| Host | Digital Ocean ($6/mo) |
| Domain | jswan.dev |
| Engine | Docker + Caddy |
| Storage | SQLite + Local Disk |
Setup was straightforward. The official Bluesky installer handles most of it—sets up Docker, configures Caddy for TLS certificates, gets everything running. Main work was configuring DNS records (A record for root domain, wildcard for subdomains) and running through installer prompts.

Once it was up, I created an account: @com.jswan.dev. Handle format is username.yourdomain.com.

After going through the setup, I realized there wasn’t a decent guide that walked through everything start to finish. So I made one.
It covers:
Written for people who are technical but maybe haven’t self-hosted much.
I have a Bluesky account at @com.jswan.dev, and I’ve set up this whole infrastructure, but I’m not really posting on social media. Not my thing.
But that’s fine. The point wasn’t necessarily to become an active Bluesky user. The point was learning how federated protocols work, understanding how to set up and maintain a service, and having the infrastructure if I want it. If friends want accounts, I can give them invites. The data lives on my server, even if that data is currently just me posting a sweet link to a swantron blog post once every three months.
Creating the guide was valuable—forced me to think through the process clearly and make it reproducible. Hopefully it saves somebody debug time.
Running your own PDS:
Total: around $100/year.
If you’re interested in setting up your own PDS, check out the guide: github.com/swantron/bluesky-pds-guide.
Self-host your identity. It’s cheap and probably a good thing to do.
This is the first time in decades I haven’t had a WordPress instance live. Lordamercy…

Over two decades, swantron.com hopped through several hosting trends:
Each previous phase was just a different way of babysitting a server. This migration is different. It’s not just a new host; it’s a fundamental change in philosophy from ‘managed system’ to code-first delivery.
I had a shower thought about how gross a WordPress database might end up after being left out in the rain for 20 years (2005-2025). I finally checked. My final database dump: 34MB and 555,947 lines of text.

bouncerblog.com—a domain that died over a decade ago.wp_options._transient entries and orphaned plugin settings that WordPress autoloads on every single page request, long after the plugins are deleted.The database had become a dumpster fire. Transitioning to Markdown files is nice. No more regex-searching a SQL dump just to find a setting.
Moving 1,040 posts is a special kind of hell. The goal was to strip away the ugly legacy /index.php/ prefix from my URLs without breaking 20 years of external links and search indexing.
I wrote some dirty Python to automate the alias field in Hugo’s spec to map the shitty legacy paths to the new clean ones.
The result in each Markdown file:
title: "This Old Post about Hot Sauce"
slug: "this-old-post-about-hot-sauce"
aliases:
- /index.php/2005/10/10/this-old-post-about-hot-sauce/

Super straightforward. Hugo generates redirect HTML pages at the old paths during the build. This preserves every bookmark (yeah right), share, and search result while allowing the site to live at a modern URL.
Result: 1,041/1,041 posts migrated with 100% link integrity.
Hugo 0.154.5 for static generation, GitHub Pages + Actions for hosting and CI/CD. No themes—just custom CSS and layout code that I control entirely. No comments, because I’m not collecting feedback from blog commentators.
| Feature | WordPress | Hugo |
|---|---|---|
| Speed | 2s - 4s Load | < 500ms |
| Security | Constant Patches | Zero Attack Surface |
| Cost | Monthly Fees | $0 (GitHub Pages) |
| URL Structure | /index.php/slug | Sane |
| Content | MySQL | Markdown / Git |
I write Markdown, I git push, and it’s live.
It feels like the OG blogging days again. We had CMS shit back then, but blogging was largely just writing and publishing. Simple and direct. SEO made things weird for a bit (paid posts… did that) and we tried to mash PHP junk onto all sorts of places for no particular reason. The CMS never really got better, and blogs sort of died under their own weight.
In a way, I’m jumping back in with the terminal Gs who never bothered down this path in the first place. They’ve been over there in their minimal setups, posting random shit about obsolete tech this entire time, while the rest of us were fighting database corruption.

Here’s to another decade, though… this time it is static, versioned, and finally cattle, not pets.
12 year old provided bath bomb to 14 year old
(12 year old is in the ET program and was the 6th grade spelling bee runner up)

Do you hate cooking blogs? Sure, we all do..
Inane story, some ads, ingredient somewhere, more adds, quatities somewhere else, another ad. Like and subscribe..
It is embarrassing to say that I bought the domain name with that sort of thing in mind. I have some of our go-tos in texts from Katie and others that I have emailed to myself (from texts from Katie). I parked some stuff on WordPress and promptly remembered that I hate WordPress about as much as I hate cooking blogs. We know what we have and what we like—it wasn’t a very practical idea. No reason to host that sort of thing.
So instead, I sort of built an anti-food-blog: Chomptron. It’s an AI-powered engine that turns whatever’s in your fridge into actual recipes. No fluff, no life stories, just dinner.

You’ve got chicken, some tomatoes, garlic, and half an onion. Or maybe just sriracha and some leftovers. You don’t want to read an article; you just want to know how to combine those things into something edible without making a grocery run.
Type the ingredients, hit generate, and get a recipe with scaled measurements and instructions. It uses Google Gemini under the hood to handle the logic. Bam..
Features that actually matter:
localStorage. I don’t want your email, I don’t want your data, and I don’t want to manage a user databaseI wanted this to be fast and ’tidy.’ No framework bloat, no heavy lifting on the client side:
gemini-2.5-flash-lite)Chomptron runs on Google Cloud Run, which fits the old ‘cattle, not pets’ thing nicely:
I built in some smart caching (24-hour TTL) and rate-limiting to keep the Gemini API costs under control, but otherwise, it’s a hands-off deployment.
The deployment cycle is refreshing. Push to main triggers a Cloud Build which handles the Dockerization and pushes to Cloud Run. It’s a gang of tests and a few seconds of waiting before it’s live.
It’s got the regular dev junk:
/health, /ready)Live: https://chomptron.com
Source: https://github.com/swantron/chomptron
It’s free to use and nearly free to run (plz don’t spam it). It was a fun excuse to get back into GCP and keep a JS project clean. No bloat, no stories—just the recipes. Chef kiss..
Do you hate insecure base images? Sure, we all do..
I built a thing: secure-base-images. It’s a minimal, security-hardened Docker base image for static Go binaries.

Most Docker images are bloated. They include shells, package managers, and a ton of dependencies you don’t need. This creates a huge attack surface that your security team loves to talk about, and drives fast pipeline guys like me insane. For static Go tools, you literally just need the binary and some certs.
A distroless base image that gives you:
FROM swantron/secure-base:latest
COPY myapp /app
ENTRYPOINT ["/app"]
That’s it. Push a tag, GitHub Actions builds it, scans it with Trivy, and publishes to Docker Hub if it’s clean (it is.)

The GitHub Actions workflow is doing the heavy lifting:
It’s opinionated but in a good way. Security by default.

Source: https://github.com/swantron/secure-base-images
Docker Hub: swantron/secure-base:latest
The QUICKSTART.md gets you from zero to published in about 10 minutes.
I’ve been busy with tron swan dot com.

It’s still just hammering on stuff and learning, with a robot motif. The robot spins.

Reminder: the site is a playground, and is sort of stupid. It is also a nicely done React site with modern patterns and a legit pipeline. I run a bunch of health checks for other services.. there is a nice weather feature.. I use it to stream Spotify while I work. It is all over the place.

Here’s the source: https://github.com/swantron/tronswan
strain noodles: ❌
strain sun: ✅
40%-ish solar eclipse in Bozeman. Time to make crescents.
10 year-old provided jellybean feedback to 12 year-old..

Alex from Peloton likes to say something along the lines of “you don’t have to get ready if you stay ready.” Its pretty good advice.
Along those lines I decided to put together a react app to stay fresh. I’ve been hammering on (gitlab) pipelines and api frameworks at work for most of the year.. time for a project.
It is live on tronswan.com
The goals were to:
- hit some APIs from react
- mess around and build some components
- implement playwright on a project
- write GH Actions to handle build / test / deploy
- POC CI/CD to DigialOcean
Pretty fun project.. I’m displaying weather stuff and doing fizzbuzz via a weird component. Full CI/CD. Take a look if you’re bored.