How to migrate from stable to unstable NixOS mid-cycle
| OS / Distro | NixOS |
|---|---|
| Category | Operating Systems |
| Guide type | Procedure |
| Skill level | Intermediate to advanced |
| Time | 15 - 60 minutes including verification |
When How to migrate from stable to unstable NixOS mid-cycle bites you on NixOS, the first instinct is to open a support ticket. Most of the time you do not have to. The steps below are the ones a senior Linux engineer would walk you through on a war-room call.
What how to migrate from stable to unstable nixos mid-cycle actually involves on NixOS
This task on NixOS is one of the more searched operational topics across distro forums and Unix StackExchange in the last 12 months. The procedure below is the path that works on a current NixOS install with default config.
The rest of this page is the structured fix path. Start with diagnose, then remediation, then the automation options so you do not have to do this by hand the next time it surfaces. Verify and safety sections at the end are the discipline that keeps the fix from regressing in production.
Diagnose first, fix second
Check service journal for the calling unit. journalctl -u <service> --since today --no-pager shows the full unit timeline. Add -p err to filter to errors only. Use journalctl -u <service> -f in another terminal while you reproduce; the bug usually surfaces in the live log within seconds.
Pull the kernel ring buffer with dmesg --since '5 minutes ago' for hardware-level events, and journalctl --since '5 minutes ago' --no-pager for the systemd timeline of the same window. Cross-reference them. Most boot, network, and storage issues on {family} leave a signature in both at the same wall-clock timestamp.
Check the vendor status page and any release-notes feed before assuming the issue is local. Distro security advisories from Ubuntu USN, Debian DSA, RHEL Errata, SUSE SU, and Arch security tracker often warn about a known regression within hours. About one in ten user-reported breakages turns out to be a known recent change already tracked upstream.
Solution-focused remediation path
If storage is suspect, capture both the block-device view and the filesystem view. lsblk -f + blkid + df -hT + du -shx /* + findmnt + mount | column -t. For ZFS use zpool status -v and zfs list -t snapshot. For Btrfs use btrfs filesystem usage / and btrfs subvolume list /. About a third of 'disk full' issues on Btrfs are metadata exhaustion, where df shows free space but the filesystem refuses writes.
Most NixOS failures fall into one of three buckets: configuration drift (a setting changed and nobody documented it), dependency gap (a package, kernel module, or library is missing or wrong version), or resource exhaustion (disk, memory, file handles, or inodes). Triage in that order. It covers around 80 percent of real-world cases. If the failure does not fit any of the three, it is likely an upstream regression worth tracking against the distro bug tracker.
For boot issues, the right primitive is the rescue console. UEFI dropdown to the firmware setup, boot from the install ISO, mount the root filesystem, and chroot into it. Once chrooted you can reinstall the bootloader (grub-install + update-grub on Debian family, grub2-install + grub2-mkconfig on RHEL family, bootctl install for systemd-boot), regenerate initramfs (update-initramfs -u -k all, dracut --force --regenerate-all, mkinitcpio -P), and reset the root password (passwd).
Automate this fix so you do not do it twice
Automate the fix in shell with systemctl, journalctl, and the package manager
On most Linux and BSD systems the most reliable repair primitives are the built-in CLI tools. systemctl status reveals the current service state, journalctl -u exposes the structured log stream, and systemctl reload or restart applies config changes without a reboot. For package management use the distro tool: apt, dnf, zypper, pacman, pkg, opkg, apk. For hardware and inventory checks the canonical readers are lsblk, lspci, lscpu, dmidecode, and lsmod.
# Template - replace SERVICE with the failing unit name
systemctl status SERVICE --no-pager | head -40
journalctl -u SERVICE -n 100 --no-pager
ss -tlnp | grep -i SERVICE
ls -l /etc/SERVICE/ 2>/dev/null
cat /etc/os-releaseWire the fix into a systemd unit override or Ansible role for self-healing
If the underlying cause is a setting that drifts over time, do not script the fix repeatedly. Bake it into a configuration-management role that runs on every check-in. Ansible, Puppet, Chef, SaltStack, and tools like Cockpit, Foreman, and Spacewalk all support enforced state. The role reasserts itself, so even if an operator changes the setting locally, the next run brings it back to the codified state (typically every 30 minutes for Puppet, on cron or systemd-timer for Ansible).
# Ansible task that enforces the corrected setting on every run
- name: Enforce hardened sshd config ansible.builtin.lineinfile: path: /etc/ssh/sshd_config regexp: '^#?PermitRootLogin' line: 'PermitRootLogin no' backup: yes notify: restart sshdCodify the fix as a systemd timer or cron job for unattended remediation
For workflows that need to run unattended (clear a stuck cache, rotate logs, fail over a service, rebuild an index) a systemd timer or a cron job is the right place. Timers can fire on boot, on schedule, or after a dependency unit reaches an active state. systemctl list-timers shows the next-fire time for every active timer. For interactive helper workflows, a wrapper shell script in /usr/local/bin/ documented in MOTD or the team wiki keeps the institutional knowledge accessible.
Common pitfalls and what to watch for
The most common pitfall when fixing this on NixOS is treating it as a one-off rather than as a recurring class of incident. The same misconfiguration tends to happen again after a kernel upgrade, a major distro version bump, or a fleet rollout unless the fix is codified. Add an Ansible role, a Puppet manifest, a SaltStack state, or a Cloud-init drop-in that prevents the same misconfig from being reintroduced. Documentation alone does not survive team turnover.
Another common trap: confirming the fix on a single host and assuming the fleet is healthy. Loop your check across every node, container, and VM that could exhibit the same symptom. If you cannot enumerate the affected scope without a script, you do not yet understand the scope.
Verify the fix worked
- Reproduce the original symptom path. If it still surfaces on any host, container, or VM in the fleet, you have not fixed it.
- Watch for 24 to 48 hours.
journalctl --since '24 hours ago' -u <service> -p errand Prometheus query history can mask issues with cached health for 6 to 12 hours, especially for slow-burn memory leaks and disk-fill regressions. - Run a smoke test under realistic load. Happy-path tests miss race conditions, file-descriptor leaks, and cgroup limits.
- Capture the new state in a runbook so the next person on call does not have to rediscover this. Push it to Confluence or your team wiki, not into Slack.
- If the fix involved a permission or security change, run a CIS Benchmark or DISA STIG audit one more time to confirm you did not open a separate hole while closing this one.
Safety, rollback, blast radius
- Test in a non-production VM, container, or namespace if your environment supports it. The cost of one disposable VM is cheaper than one rollback meeting.
- Export the existing config before changing it. Most NixOS services support
--print-defaults,systemctl show, or a documented config-dump command. Capture that to source control before you start. - Know your rollback path. Some NixOS operations are one-way (irreversible filesystem upgrade like ext3 to ext4 inline, kernel ABI change, removal of an LVM physical volume). Confirm reversibility on the official OS documentation before you commit.
- Be aware of cross-service impact. A change to PAM ripples to every service using it. A change to /etc/resolv.conf affects every name lookup. A change to systemd default.target affects every reboot.
- Maintenance window discipline: if the change touches DNS, certificate rotation, kernel upgrade, or anything that emits TLS handshakes, line up a window with stakeholder notification, not a heroic mid-day swap.
FAQ
etckeeper commit, cp file file.bak.$(date +%F), or a Btrfs/ZFS snapshot), then commit it before you change anything. A few operations are one-way (in-place filesystem conversion, partition table rewrite, kernel ABI bump). Check the distro release notes for the specific operation before you commit.systemctl list-dependencies and lsof to enumerate consumers before changing a shared service or configuration file.man <command> on the host, or the upstream project documentation - those almost always still work.sosreport (RHEL family) or supportconfig (SUSE), and your reproduction steps. The distro forum is the no-cost public alternative - search there first; 80 percent of common NixOS issues already have a working answer marked as solved.References
- Official documentation for NixOS
- Distro forums and community Q&A (Ubuntu Discourse, Fedora Discussion, Arch BBS, openSUSE Forum, Reddit r/linux + distro subreddits, ServerFault, Unix StackExchange)
- Vendor status pages and release-notes feeds
- CIS Benchmarks and DISA STIG hardening guides for NixOS
Related fixes
Related guides worth a look while you sort this one out:
- How to switch Manjaro between Stable Testing Unstable branches
- How to roll back from Alpine edge to stable
- How to migrate from Amazon Linux 2 to AL2023 missing packages
- How to migrate from old watch to new Wear OS watch
- How to migrate CentOS Stream 8 to CentOS Stream 9 with leapp
- How to migrate from CentOS Stream to Rocky or AlmaLinux