5.3 KiB
nixos-rebuild-ng
Work-in-Progress rewrite of
nixos-rebuild
.
Why the rewrite?
The current state of nixos-rebuild
is dare: it is one of the most critical
piece of code we have in NixOS, but it has tons of issues:
- The code is written in Bash, and while this by itself is not necessary bad, it means that it is difficult to do refactorings due to the lack of tooling for the language
- The code itself is a hacky mess. Changing even one line of code can cause issues that affects dozens of people
- Lack of proper testing (we do have some integration tests, but no unit tests and coverage is probably pitiful)
- The code predates some of the improvements
nix
had over the years, e.g.: it builds Flakes inside a temporary directory and read the resulting symlink since the code seems to predate--print-out-paths
flag
Given all of those above, improvements in the nixos-rebuild
are difficult to
do. A full rewrite is probably the easier way to improve the situation since
this can be done in a separate package that will not break anyone. So this is
an attempt of the rewrite.
Why Python?
- It is the language of choice for many critical things inside
nixpkgs
, like theNixOSTest
andsystemd-boot-builder.py
activation scripts - It is a language with great tooling, e.g.:
mypy
for type checking,ruff
for linting,pytest
for unit testing - It is a scripting language that fits well with the scope of this project
- Python's standard library is great and it means we will need a low number of
external dependencies for this project. For example,
nixos-rebuild
currently depends injq
for JSON parsing, while Python hasjson
in standard library
Do's and Don'ts
- Do: be as much of a drop-in replacement as possible
- Do: fix obvious bugs
- Do: improvements that are non-breaking
- Don't: change logic in breaking ways even if this would be an improvement
How to use
{ pkgs, ... }:
{
environment.systemPackages = [ pkgs.nixos-rebuild-ng ];
}
And use nixos-rebuild-ng
instead of nixos-rebuild
.
Development
Run:
nix-build -A nixos-rebuild-ng.tests.ci
The command above will run the unit tests and linters, and also check if the code is formatted. However, sometimes is more convenient to run just a few tests to debug, in this case you can run:
nix-shell -A nixos-rebuild-ng.devShell
The command above should automatically put you inside src
directory, and you
can run:
# run program
python -m nixos_rebuild
# run tests
pytest
# check types
mypy .
# fix lint issues
ruff check --fix .
# format code
ruff format .
Current caveats
- For now we will install it in
nixos-rebuild-ng
path by default, to avoid conflicting with the currentnixos-rebuild
. This means you can keep both in your system at the same time, but it also means that a few things like bash completion are broken right now (since it looks atnixos-rebuild
binary) _NIXOS_REBUILD_EXEC
is not implemented yet, so different fromnixos-rebuild
, this will use the current version ofnixos-rebuild-ng
in yourPATH
to build/set profile/switch, whilenixos-rebuild
builds the new version (the one that will be switched) and re-exec to it instead. This means that in case of bugs innixos-rebuild-ng
, the only way that you will get them fixed is after you switch to a new versionnix
bootstrap is also not implemented yet, so this means that you will eval with an old version of Nix instead of a newer one. This is unlikely to cause issues, because the build will happen in the daemon anyway (that is only changed after the switch), and unless you are using bleeding edgenix
features you will probably have zero problems here. You can basically think that usingnixos-rebuild-ng
is similar to runningnixos-rebuild --fast
right now- Ignore any performance advantages of the rewrite right now, because of the 2 caveats above
--target-host
and--build-host
are not implemented yet and this is probably the thing that will be most difficult to implement. Help here is welcome- Bugs in the profile manipulation can cause corruption of your profile that
may be difficult to fix, so right now I only recommend using
nixos-rebuild-ng
if you are testing in a VM or in a filesystem with snapshots like btrfs or ZFS. Those bugs are unlikely to be unfixable but the errors can be difficult to understand. If you want to go anyway,nix-collect-garbage -d
andnix store repair
are your friends
TODO
- Remote host/builders (via SSH)
- Improve nix arguments handling (e.g.:
nixFlags
vscopyFlags
in the oldnixos-rebuild
) _NIXOS_REBUILD_EXEC
- Port
nixos-rebuild.passthru.tests
- Change module system to allow easier opt-in, like
system.switch.enableNg
forswitch-to-configuration-ng
- Improve documentation
nixos-rebuild repl
(calling oldnixos-rebuild
for now)nix
build/bootstrap- Generate tab completion via
shtab
- Reduce build closure
TODON'T
- Reimplement
systemd-run
logic (will be moved to the newapply
script)