Install NixOS on Hetzner Cloud: Difference between revisions
Oldhomemovie (talk | contribs) mNo edit summary |
Fix nixos-anywhere install |
||
(29 intermediate revisions by 11 users not shown) | |||
Line 9: | Line 9: | ||
There are several ways to install NixOS, such as the "traditional" ISO installation, [[nixos-infect]] or [[nixos-anywhere]]. | There are several ways to install NixOS, such as the "traditional" ISO installation, [[nixos-infect]] or [[nixos-anywhere]]. | ||
=== | === Traditional ISO installation === | ||
In the Hetzner console, mount the NixOS minimal ISO into your server, and use the console to install NixOS. | |||
==== x86_64 ==== | |||
At time of writing, Hetnzer's x86_64 servers use legacy boot. | |||
First, create a new MBR partition table. | |||
parted /dev/sda --script mklabel msdos | |||
Then create a 512MB boot partition with ext4 | |||
parted /dev/sda --script mkpart primary ext4 1MiB 513MiB | |||
parted /dev/sda --script set 1 boot on | |||
mkfs.ext4 -L boot /dev/sda1 | |||
Create a swap partition. This example uses 8GB, you may want to research the correct amount for your server size. Note the end of the swap partition in this command is 8577MiB, this is the value used in the next command. | |||
parted /dev/sda --script mkpart primary linux-swap 513MiB 8577MiB | |||
mkswap -L swap /dev/sda2 | |||
swapon /dev/sda2 | |||
Create a root partition using the rest of the disk with ext4. | |||
parted /dev/sda --script mkpart primary ext4 8577MiB 100% | |||
mkfs.ext4 -L nixos /dev/sda3 | |||
If you don't mount the partitions you've just created, the NixOS installer will produce an error in the form `Failed to get blkid info (returned 512) for on tmpfs at <path>/<prefix>-install-grub.pl`. | |||
# Mount the partitions to /mnt and /mnt/boot. | |||
mount /dev/disk/by-label/nixos /mnt | |||
mkdir /mnt/boot | |||
mount /dev/disk/by-label/boot /mnt/boot | |||
Finally, install. Install from a remote flake: | |||
sudo nixos-install --flake github:<username>/<repo>#<id> | |||
Once installed, unmount the ISO and reboot. | |||
===== Hetnzer base configuration ===== | |||
In the example below, the id would be `hetzner-x86_64`. | |||
The `flake.nix` file in the repo should be of the form: | |||
{ | |||
inputs = { | |||
nixpkgs.url = "github:NixOS/nixpkgs/nixos-24.11"; | |||
}; | |||
outputs = { nixpkgs, ... }: { | |||
nixosConfigurations = { | |||
hetzner-x86_64 = nixpkgs.lib.nixosSystem { | |||
system = "x86_64-linux"; | |||
modules = [ | |||
./configuration.nix | |||
]; | |||
}; | |||
}; | |||
}; | |||
} | |||
With the referenced `configuration.nix` in the form. | |||
Note the filesystems configuration, which matches the partition scheme, and the `availableKernelModules` section which includes modules that enable ext4 at boot. | |||
Also note the user `username` which is configured to be part of the `wheel` group, and can therefore use `sudo`. | |||
{ pkgs, ... }: | |||
{ | |||
nix.settings = { | |||
experimental-features = "nix-command flakes"; | |||
}; | |||
environment.systemPackages = [ | |||
pkgs.vim | |||
pkgs.git | |||
]; | |||
fileSystems."/" = { | |||
device = "/dev/disk/by-label/nixos"; | |||
fsType = "ext4"; | |||
}; | |||
fileSystems."/boot" = { | |||
device = "/dev/disk/by-label/boot"; | |||
fsType = "ext4"; | |||
}; | |||
swapDevices = [ | |||
{ | |||
device = "/dev/disk/by-label/swap"; | |||
} | |||
]; | |||
time.timeZone = "Europe/London"; | |||
i18n.defaultLocale = "en_US.UTF-8"; | |||
console.keyMap = "us"; | |||
boot.loader.grub.enable = true; | |||
boot.loader.grub.device = "/dev/sda"; | |||
boot.initrd.availableKernelModules = [ "ahci" "xhci_pci" "virtio_pci" "virtio_scsi" "sd_mod" "sr_mod" "ext4" ]; | |||
users.users = { | |||
root.hashedPassword = "!"; # Disable root login | |||
username = { | |||
isNormalUser = true; | |||
extraGroups = [ "wheel" ]; | |||
openssh.authorizedKeys.keys = [ | |||
''ssh-rsa <your_ssh_public_key>'' | |||
]; | |||
}; | |||
}; | |||
security.sudo.wheelNeedsPassword = false; | |||
services.openssh = { | |||
enable = true; | |||
settings = { | |||
PermitRootLogin = "no"; | |||
PasswordAuthentication = false; | |||
KbdInteractiveAuthentication = false; | |||
}; | |||
}; | |||
networking.firewall.allowedTCPPorts = [ 22 ]; | |||
system.stateVersion = "24.11"; | |||
} | |||
To access the VM, you will need to ensure that port 22 on the VM is opened via the Hetzner firewall if that is configured. | |||
=== nixos-anywhere === | |||
The tutorial assumes you already have an account on Hetzner Cloud, and no prior access to a system with NixOS or nix CLI utility installed. | |||
#First upload your SSH key via the Hetzner Web UI | |||
#Then click yourself a VM. For the OS choose Ubuntu but anything should work. This guide was tested with x86_64-linux but aarch64 should work with the note from below. | |||
#Using a code editor on your host computer, create 4 files. File contents, as well as the location of where to put corresponding file are indicated below:<syntaxhighlight lang="nix"> | #Using a code editor on your host computer, create 4 files. File contents, as well as the location of where to put corresponding file are indicated below:<syntaxhighlight lang="nix"> | ||
# /tmp | # /tmp/my-hetzner-vm/hardware-configuration.nix | ||
{ config, lib, pkgs, modulesPath, ... }: | { config, lib, pkgs, modulesPath, ... }: | ||
Line 47: | Line 158: | ||
]; | ]; | ||
networking.useDHCP = lib.mkDefault true; | networking.useDHCP = lib.mkDefault true; | ||
nixpkgs.hostPlatform = lib.mkDefault "x86_64-linux"; | nixpkgs.hostPlatform = lib.mkDefault "x86_64-linux"; | ||
} | } | ||
</syntaxhighlight><syntaxhighlight lang="nix"> | </syntaxhighlight><syntaxhighlight lang="nix"> | ||
# /tmp | # /tmp/my-hetzner-vm/disko-config.nix | ||
{ | { | ||
Line 96: | Line 202: | ||
} | } | ||
</syntaxhighlight><syntaxhighlight lang="nix"> | </syntaxhighlight><syntaxhighlight lang="nix"> | ||
# /tmp | # /tmp/my-hetzner-vm/configuration.nix | ||
{ config, lib, pkgs, ... }: | { config, lib, pkgs, ... }: | ||
Line 120: | Line 226: | ||
enable = true; | enable = true; | ||
defaultEditor = true; | defaultEditor = true; | ||
}; | }; | ||
system.stateVersion = "24. | system.stateVersion = "24.11"; | ||
} | } | ||
</syntaxhighlight>'''Note''': the value of <code>initialHashedPassword</code> was obtained using <code>mkpasswd</code> command in Linux, and corresponds to <code>Password.123</code> string used as password.<syntaxhighlight lang="nix"> | </syntaxhighlight>'''Note''': the value of <code>initialHashedPassword</code> above was obtained using <code>mkpasswd</code> command in Linux, and corresponds to <code>Password.123</code> string used as password.<syntaxhighlight lang="nix"> | ||
# /tmp/my- | # /tmp/my-hetzner-vm/flake.nix | ||
{ | { | ||
inputs = { | inputs = { | ||
nixpkgs = { | nixpkgs = { | ||
url = "github:NixOS/nixpkgs/nixos-24. | url = "github:NixOS/nixpkgs/nixos-24.11"; | ||
}; | }; | ||
Line 160: | Line 255: | ||
modules = [ | modules = [ | ||
. | ./configuration.nix | ||
inputs.disko.nixosModules.disko | inputs.disko.nixosModules.disko | ||
]; | ]; | ||
Line 167: | Line 262: | ||
}; | }; | ||
} | } | ||
</syntaxhighlight> | </syntaxhighlight> | ||
#To build NixOS from the flake run:<syntaxhighlight lang="shell"> | |||
nix run --extra-experimental-features 'nix-command flakes' github:nix-community/nixos-anywhere -- --flake /tmp/my-hetzner-vm#my-hetzner-vm --target-host root@0.0.0.0 --build-on-remote | |||
</syntaxhighlight>'''Note''': replace <code>0.0.0.0</code> with an IP address obtained during an earlier step. | |||
The NixOS on Hetzner is installed! | |||
=== disko === | === disko === | ||
Line 215: | Line 274: | ||
references: | references: | ||
* [[Disko]] | |||
* [https://github.com/feelssexy/hetzner-auto-nixos/blob/main/hardware-configuration.nix sample regular hardware config] | * [https://github.com/feelssexy/hetzner-auto-nixos/blob/main/hardware-configuration.nix sample regular hardware config] | ||
* [https://github.com/LGUG2Z/nixos-hetzner-cloud-starter/blob/master/disk-config.nix sample config using disko] | * [https://github.com/LGUG2Z/nixos-hetzner-cloud-starter/blob/master/disk-config.nix sample config using disko] | ||
Line 233: | Line 293: | ||
* Run following script. Replace <code>NIX_CHANNEL</code> variable with the version string you wish to install. | * Run following script. Replace <code>NIX_CHANNEL</code> variable with the version string you wish to install. | ||
<syntaxHighlight lang=bash> | <syntaxHighlight lang=bash> | ||
curl https://raw.githubusercontent.com/elitak/nixos-infect/master/nixos-infect | NIX_CHANNEL=nixos- | curl https://raw.githubusercontent.com/elitak/nixos-infect/master/nixos-infect | NIX_CHANNEL=nixos-24.11 bash -x | ||
</syntaxHighlight> | </syntaxHighlight> | ||
* Reboot into NixOS | * Reboot into NixOS | ||
Line 248: | Line 308: | ||
<syntaxhighlight lang="nix"> | <syntaxhighlight lang="nix"> | ||
systemd.network.enable = true; | |||
systemd.network.networks."30-wan" = { | |||
matchConfig.Name = "ens3"; # either ens3 or enp1s0, check 'ip addr' | |||
networkConfig.DHCP = "ipv4"; | |||
address = [ | |||
# replace this subnet with the one assigned to your instance | |||
"2a01:4f8:aaaa:bbbb::1/64" | |||
]; | |||
routes = [ | |||
{ Gateway = "fe80::1"; } | |||
]; | |||
}; | |||
</syntaxhighlight> | </syntaxhighlight> | ||
Line 264: | Line 326: | ||
<syntaxhighlight lang="nix"> | <syntaxhighlight lang="nix"> | ||
systemd.network.networks."30-wan" = { | |||
matchConfig.Name = "ens3"; # either ens3 (amd64) or enp1s0 (arm64) | |||
networkConfig.DHCP = "no"; | |||
address = [ | |||
# replace this address with the one assigned to your instance | |||
"A.B.C.D/32" | |||
# replace this subnet with the one assigned to your instance | |||
"2a01:4f8:AAAA:BBBB::1/64" | |||
]; | |||
routes = [ | |||
{ Gateway = "172.31.1.1"; GatewayOnLink = true; } | |||
{ Gateway = "fe80::1"; } | |||
]; | |||
}; | |||
</syntaxhighlight> | </syntaxhighlight> | ||
Line 290: | Line 352: | ||
[[Category:Cookbook]] | [[Category:Cookbook]] | ||
[[Category:Deployment]] |