-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathnetboot.nix
129 lines (114 loc) · 3.37 KB
/
netboot.nix
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
# mostly taken from https://github.com/cleverca22/nixos-configs/blob/master/netboot_server.nix
{ lib, config, pkgs, ... }:
let
nixos_release = import (pkgs.path + "/nixos/release.nix") {};
netboot = let
build = (import (pkgs.path + "/nixos/lib/eval-config.nix") {
modules = [
(pkgs.path + "/nixos/modules/installer/netboot/netboot-minimal.nix")
cfg.image
];
}).config.system.build;
in pkgs.symlinkJoin {
name = "netboot";
paths = with build; [ netbootRamdisk kernel netbootIpxeScript ];
};
ipxe' = pkgs.ipxe.overrideDerivation (drv: {
installPhase = ''
${drv.installPhase}
make $makeFlags bin-x86_64-efi/ipxe.efi bin-i386-efi/ipxe.efi
cp -v bin-x86_64-efi/ipxe.efi $out/x86_64-ipxe.efi
cp -v bin-i386-efi/ipxe.efi $out/i386-ipxe.efi
'';
});
tftp_root = pkgs.runCommand "tftproot" {} ''
mkdir -pv $out
cp -vi ${ipxe'}/undionly.kpxe $out/undionly.kpxe
cp -vi ${ipxe'}/x86_64-ipxe.efi $out/x86_64-ipxe.efi
cp -vi ${ipxe'}/i386-ipxe.efi $out/i386-ipxe.efi
'';
nginx_root = pkgs.runCommand "nginxroot" {} ''
mkdir -pv $out
cat <<EOF > $out/boot.php
#!ipxe
chain netboot/netboot.ipxe
EOF
ln -sv ${netboot} $out/netboot
'';
cfg = config.netboot;
in {
options = {
netboot = {
network.wan = lib.mkOption {
type = lib.types.str;
description = "the internet facing IF";
};
network.lan = lib.mkOption {
type = lib.types.str;
description = "the netboot client facing IF";
};
image = lib.mkOption {
type = lib.types.path;
description = "the image to serve";
};
};
};
config = {
services = {
nginx = {
enable = true;
virtualHosts = {
"192.168.3.1" = {
root = nginx_root;
};
};
};
dhcpd4 = {
interfaces = [ cfg.network.lan ];
enable = true;
extraConfig = ''
option arch code 93 = unsigned integer 16;
subnet 192.168.3.0 netmask 255.255.255.0 {
option domain-search "localnetboot";
option subnet-mask 255.255.255.0;
option broadcast-address 192.168.3.255;
option routers 192.168.3.1;
option domain-name-servers 192.168.3.1, 8.8.8.8, 8.8.4.4;
range 192.168.3.100 192.168.3.200;
next-server 192.168.3.1;
if exists user-class and option user-class = "iPXE" {
filename "http://192.168.3.1/boot.php?mac=''${net0/mac}&asset=''${asset:uristring}&version=''${builtin/version}";
} else {
if option arch = 00:07 or option arch = 00:09 {
filename = "x86_64-ipxe.efi";
} else {
filename = "undionly.kpxe";
}
}
}
'';
};
tftpd = {
enable = true;
path = tftp_root;
};
bind = {
enable = true;
cacheNetworks = [ "192.168.3.0/24" "127.0.0.0/8" ];
};
};
networking = {
interfaces = {
${cfg.network.lan} = {
ipv4.addresses = [ { address = "192.168.3.1"; prefixLength = 24; } ];
};
};
nat = {
enable = true;
externalInterface = cfg.network.wan;
internalIPs = [ "192.168.3.0/24" ];
internalInterfaces = [ cfg.network.lan ];
};
};
};
}