Nix as a Homebrew Replacement (A Failed Attempt)
Table of Contents
Why Nix?
For the past few months, I’ve been using Nix (the package manager, not the OS) as my package manager across multiple devices (MacBooks, Raspberry Pi, Linux workstations). My goals don’t align with Nix’s visions perfectly, but compared to Homebrew (and Apt), it does offer quite a few advantages:
Configure once, run anywhere: Homebrew’s support for Linux is very basic and flaky, and doesn’t work for Raspberry Pi. Nix works across all my systems, macOS (both Intel and Apple Silicon), ARM64 (Raspberry Pi), and x86 Linux workstations.
Declarative package management: I can check in my Nix configurations in a Git repo sync’ed across multiple devices, and apply them in a single command. This also means that removing packages is much cleaner as you don’t need to worry about unneeded dependencies (like apt autoremove
or brew rmtree
)1.
Supports Python packages: I can declare (python3.withPackages (ps: with ps; [ ipython pip pyyaml requests strictyaml plotly pandas packaging dash scipy ]))
which saves a lot of trouble maintaining Python environments.
A large package registry: Once I spent hours trying to build autotrace
on my Raspberry Pi, finding out it is as simple as nix-env -i autotrace
was an instant-switch for me. You can find most (if not all) packages you need on Nix Search.
My Requirements
I need:
- Single file: A single declarative config file that I can check into my version control monorepo, alongside all my other configuration files like Neovim and Hammerspoon’s init.lua
- Run everywhere: It should work for all my OS and devices
- No boilerplate: It should have as little boilerplate as possible: managing packages can’t be that complicated, right?
- One command to apply: I should be able to apply the said config file with a single command with no human intervention (i.e.
git pull && apply-nix-env
should do it)
(Note: Notice how being “reproducible” is not one of them, despite being one of Nix’s major selling points. This is because personally I’ve never ran into reproducibility issues, at least with Homebrew.)
My Workflow
Due to my specific requirements, my Nix workflow might not align with Nix’s visions either (if there exists such a vision). 2 and 3 ruled out nix-darwin, and I did the simplest setup possible (thanks to Stephen Checkoway):
First have a env.nix
file like this:
{ pkgs ? import
(fetchTarball {
url = "https://github.com/NixOS/nixpkgs/archive/51ae8856ae59d3724c6c83e81c2854451ca7732d.tar.gz";
})
}:
with pkgs; [
(python3.withPackages (ps: with ps; [ ipython pip pyyaml requests strictyaml plotly pandas packaging dash scipy ]))
btop
cmake
...
]
Apply it with nix-env -irf env.nix
, and that’s it.
It satisfies all the requirements above, and to demonstrate the workflow I’ll speak brew
for a moment here:
brew install
becomes …
- Instead of looking it up on Homebrew Formulae, find it on Nix Search
- Edit
env.nix
and add the new package nix-env -irf
brew upgrade
becomes …
- Just update the hash in your
env.nix
(which I later automated with a script), and nix-env -irf
brew remove
becomes …
(or maybe you want brew rmtree
)
- Remove that entry from
env.nix
, and nix-env -irf
(This is a much better user experience than brew
, even with rmtree
.)
All of these operations are much faster than Homebrew (IF all binaries are available).
If-blocks
When you need it across multiple OS, or when you need some packages conditionally only on some machines, you can use lib.optional
to add if-blocks, like this:
with pkgs; [
btop
...
zsh
] ++ /* macOS */ lib.optional pkgs.stdenv.isDarwin [
blueutil
coreutils # shadows macOS commands like date, basename
diffutils # shadows macOS's BSD diff
duti
gnugrep # shadows macOS's BSD grep
gnused # shadows macOS's BSD sed
] ++ /* Linux */ lib.optional pkgs.stdenv.isLinux [
nethogs
trashy
] ++ ...
You can also add conditions on other info, e.g. lib.optional (builtins.getEnv "USER" == "hexacera")
.
How it Ends
That does satisfy all my requirements, migrating and provisioning new machines are now easier than ever, but why is this titled “(A Failed Attempt)”?
Documentation: Let’s just say that, for me, ChatGPT is Nix’s best (or should I say only) documentation. Whenever your use case deviates a little from norm, you’ll be in a hell of sprawling cryptic forum posts and trying random stuff until it works. And indeed my workflow is somewhat niche so these things happen regularly. There’s always a better use of my time than fighting with my package manager.
Freshness: I want the latest svt-av1 as each release comes with significant improvements, I don’t want to miss out on them but I’m lazy enough to not want to build from master. With Nix (nixpkgs), the version bump PRs usually comes quickly (or writing one is actually pretty easy), but it usually takes weeks for the PRs to be reviewed and merged. When PRs eventually get merged into staging
, it takes another month for them to make it to unstable
. Homebrew’s svt-av1 bottle is already 3.0.0 as of now (2025-03-08), and I’m still waiting on nixpkgs’s 3.0.0 bump to be merged.
Binary availability: With brew upgrade
or apt full-upgrade
, you know exactly what will happen: you’ll be downloading binaries and that’s it. With nix-env -irf
, once in a while I’ll find my Raspberry Pi at 100% CPU because it is trying to build Zig for some reasons. I had to script nix-env -irf
, making it do a dry-run first just to make sure it will not attempt to build the universe from source today.
Eventually I found myself migrating packages back from Nix to Homebrew.
What’s Next?
My attempt to replace Homebrew with Nix failed, but Nix is still great (even for someone that doesn’t care about reproducibility), and I still hope (and believe) Nix will get better over time, with the chicken-and-egg problems of easier onboarding → more adoption → better documentation/debuggability → easier onboarding … I’ll continue to contribute when I have the chance.
Also, let me know if you are also interested in using Nix on macOS, I have wasted so many hours making it work, and might be able to save some of yours :)
-
Though later I discovered
brew bundle
and Brewfile which offers more or less the same benefit. ↩