My Favorite Posts

XKCD's "Is It Worth the Time?" Considered Harmful — Automating things, even when it doesn't save time, is how you learn the skills to automate the things that will. (2024/12/13)

When helping elves, you should use complex numbers to navigate in 2D — Complex numbers can represent vectors in Python in a way that arrays or tuples can't, and they lead to more natural way of structuring 2D data. (2024/12/21)

Bash Patterns I Use Weekly — 4 bash tricks I find myself using often that I wish I'd discovered sooner. (2021/11/22)

Indexes are sorted arrays (sort of) — Thinking of B-trees and B+-trees as sorted arrays can help you reason about indexes and performance. (2023/08/14)

So, what can you do with a process ID? — If you have a process ID, you can look at what files/ports a process has open, see details about how the process was started, trace it to see what work it's doing, or wait for it to complete its work. Process IDs are pretty cool! (2022/12/10)

All posts

A one-table SQL query will often use a single index — For a query like WHERE A = ? AND B = ?, the most efficient index will be on (A, B) or (B, A). Two indexes on (A) and (B) won't necessarily be combined. (2025/10/08)

You should lint your TODOs — TODOs are more useful when they have a consistent format! You should decide on what types of TODOs you want to allow in your codebase and make sure they follow a consistent format that you can use for searches and scripts. (2025/09/22)

You probably want some parts of Lodash — A good chunk of the coding I've done has been munging data into appropriate shapes. It's nice to have useful helper functions that make it easier to change shapes around, and I think Lodash contains a good set of key utility functions to do that well. (2025/09/21)

Draw a Silhouette of a Spider Missing its Front Left Leg — I found it striking how incapable every AI model I tried was of drawing a spider silhouette missing one leg. (2025/06/24)

[Short gripe] Stop using .reduce to make "functional" code — I don't think I've ever seen a use of .reduce that wouldn't be clearer as either a for loop or lodash call. (2025/04/08)

Don't lecture when teaching software engineering — The basics of good pedagogy—starting with learning goals and incorporating lots of practice—can improve internal company engineer training. (2025/04/04)

I avoid easy things I haven't tried — When I haven't tried something, it ends up being a huge barrier towards adding that tool to my toolbox. It took me years to spend 2 minutes figuring out how to record my screen. (2025/04/03)

Why I avoid test stubs in TypeScript — Stubs that mutate objects are hard to maintain and can cause hard-to-debug problems for future engineers. I use "explicit stubbing" and build general test utilities whenever I need stubbing behavior. (2025/04/01)

Most engineers have git aliases; what are yours? — Git is flexible and powerful, but it takes a lot of commands to do most workflows. Most engineers I know have their own custom set of git aliases to simplify those commands. What are yours? (2025/03/26)

When helping elves, you should use complex numbers to navigate in 2D — Complex numbers can represent vectors in Python in a way that arrays or tuples can't, and they lead to more natural way of structuring 2D data. (2024/12/21)

18,957 tests in under 6 minutes: ClassDojo's approach to backend testing — When writing backend tests, the database is an integral part of what we'd like to test. Putting effort into test & fixture resetting speed can make tests that hit the database possible. (2024/12/13)

XKCD's "Is It Worth the Time?" Considered Harmful — Automating things, even when it doesn't save time, is how you learn the skills to automate the things that will. (2024/12/13)

Python patterns I learned during last year's Advent of Code — Python is a great fit for Advent of Code style problems. Here are a few patterns I picked up that I'm going to try and use again this year. (2024/11/13)

Indexes are sorted arrays (sort of) — Thinking of B-trees and B+-trees as sorted arrays can help you reason about indexes and performance. (2023/08/14)

JSDoc comments can make fixtures easier to work with — By auto-generating JSDoc comments on a fixture IDs file, we were able to make our fixture code simpler to work with. (2023/07/23)

So, what can you do with a process ID? — If you have a process ID, you can look at what files/ports a process has open, see details about how the process was started, trace it to see what work it's doing, or wait for it to complete its work. Process IDs are pretty cool! (2022/12/10)

Why does zip(*[iter(s)]*n) chunk s into n chunks in Python? — Understanding why zip(*[iter(s)]*n) chunks an input s into n chunks is a good excuse to explore iterators, iterable unpacking, list multiplication, and the zip function. (2022/12/10)

Culling Containers with a Leaky Bucket — Containers serving production traffic can sometimes get into bad states that they can't recover from. Automatically shutting down these problem containers can make production systems more resilient until you're able to debug and fix whatever the root cause is. (2022/10/24)

Setting up Image Previews using Puppeteer, Make, and CSS transforms — How I set up image previews for some posts on this site using automated screenshots. (2022/10/20)

Writing FizzBuzz in MySQL 5.7 for no good reason — MySQL 5.7 makes getting a list of numbers pretty tricky, so solutions to FizzBuzz are a fun chance to explore MySQL techniques like session variables, stored routines, and cross-joined views. (2022/07/07)

Using NodeJS's AsyncLocalStorage to Instrument a Webserver — How to start an async storage context early in your middleware stack and set up counters to track down problem routes. (2022/05/30)

Large Analytics SQL Queries are a Code Smell — Large analytics SQL queries can be unmaintainable and hard to debug. Writing large SQL queries is hard and slow! You can use temporary tables and views to break down those queries into smaller pieces that are easier to work with. (2022/05/11)

Shell Patterns for Easy Automated Code Migrations — Being comfortable using shell tools to do large code migrations is a superpower. Here are the patterns and techniques I find myself using often. (2022/05/11)

Red and Blue Function Mistakes in JavaScript — Use of async functions in JS leads to consistent mistakes; these are the ones I see most often. (2022/03/25)

Slack is the Worst Info-Radiator — When we went remote, we needed to replace the in-office info-radiators that displayed info about the health of our system. Alert slack channels are often too noisy to be useful and don't display the current state of the world well. (2021/12/06)

Bash Patterns I Use Weekly — 4 bash tricks I find myself using often that I wish I'd discovered sooner. (2021/11/22)

Canary Containers at ClassDojo in Too Much Detail — ClassDojo sends real traffic to canary containers to validate new builds and uses that as part of a system that safely deploys new code multiple times a day. (2021/11/02)

Mongo Performance is Disk-graceful — MongoDB databases often require lots of disk I/O because documents aren't stored in a clustered index and because writing snapshots can saturate disk I/O. (2021/10/22)

Node's AsyncLocalStorage Makes the Commons Legible — Node's AsyncLocalStorage can dramatically increase monolith legibility and improve performance. (2021/08/18)

The 3 things I didn't understand about TypeScript — understanding type inference, type narrowing, and structural typing is crucial to writing good TypeScript code. (2021/08/10)

Manipulating text and files solves problems — Many programming tasks can be solved with tools that manipulate text & files. Shell tools are a good fit for these tasks, but if you don't know shell tools, you can use any other programming language to accomplish the same things. (2021/08/01)

HyperLogLog-orrhea: how do HyperLogsLogs work? — HyperLogLogs leverage the power of hashing strings into random numbers to create remarkably accurate cardinality estimates. (2021/07/14)

ERROR, WARN, and INFO aren't actionable logging levels — Logs should be actionable and the default categories of FATAL, ERROR, WARN, INFO, and DEBUG aren't precise enough. (2021/07/13)

Graceful web-server shutdowns behind HAProxy — Graceful shutdowns behind a load balancer are hard to do right, but they enable solid auto-scaling and safe frequent deploys. (2021/07/12)