at 25.11-pre 9.8 kB view raw
1import ./make-test-python.nix ( 2 { pkgs, lib, ... }: 3 let 4 initiatorName = "iqn.2020-08.org.linux-iscsi.initiatorhost:example"; 5 targetName = "iqn.2003-01.org.linux-iscsi.target.x8664:sn.acf8fd9c23af"; 6 in 7 { 8 name = "iscsi"; 9 meta = { 10 maintainers = pkgs.lib.teams.deshaw.members; 11 }; 12 13 nodes = { 14 target = 15 { 16 config, 17 pkgs, 18 lib, 19 ... 20 }: 21 { 22 virtualisation.vlans = [ 23 1 24 2 25 ]; 26 services.target = { 27 enable = true; 28 config = { 29 fabric_modules = [ ]; 30 storage_objects = [ 31 { 32 dev = "/dev/vdb"; 33 name = "test"; 34 plugin = "block"; 35 write_back = true; 36 wwn = "92b17c3f-6b40-4168-b082-ceeb7b495522"; 37 } 38 ]; 39 targets = [ 40 { 41 fabric = "iscsi"; 42 tpgs = [ 43 { 44 enable = true; 45 attributes = { 46 authentication = 0; 47 generate_node_acls = 1; 48 }; 49 luns = [ 50 { 51 alias = "94dfe06967"; 52 alua_tg_pt_gp_name = "default_tg_pt_gp"; 53 index = 0; 54 storage_object = "/backstores/block/test"; 55 } 56 ]; 57 node_acls = [ 58 { 59 mapped_luns = [ 60 { 61 alias = "d42f5bdf8a"; 62 index = 0; 63 tpg_lun = 0; 64 write_protect = false; 65 } 66 ]; 67 node_wwn = initiatorName; 68 } 69 ]; 70 portals = [ 71 { 72 ip_address = "0.0.0.0"; 73 iser = false; 74 offload = false; 75 port = 3260; 76 } 77 ]; 78 tag = 1; 79 } 80 ]; 81 wwn = targetName; 82 } 83 ]; 84 }; 85 }; 86 87 networking.firewall.allowedTCPPorts = [ 3260 ]; 88 networking.firewall.allowedUDPPorts = [ 3260 ]; 89 90 virtualisation.memorySize = 2048; 91 virtualisation.emptyDiskImages = [ 2048 ]; 92 }; 93 94 initiatorAuto = 95 { 96 nodes, 97 config, 98 pkgs, 99 ... 100 }: 101 { 102 virtualisation.vlans = [ 103 1 104 2 105 ]; 106 107 services.multipath = { 108 enable = true; 109 defaults = '' 110 find_multipaths yes 111 user_friendly_names yes 112 ''; 113 pathGroups = [ 114 { 115 alias = 123456; 116 wwid = "3600140592b17c3f6b404168b082ceeb7"; 117 } 118 ]; 119 }; 120 121 services.openiscsi = { 122 enable = true; 123 enableAutoLoginOut = true; 124 discoverPortal = "target"; 125 name = initiatorName; 126 }; 127 128 environment.systemPackages = with pkgs; [ 129 xfsprogs 130 ]; 131 132 environment.etc."initiator-root-disk-closure".source = 133 nodes.initiatorRootDisk.config.system.build.toplevel; 134 135 nix.settings = { 136 substituters = lib.mkForce [ ]; 137 hashed-mirrors = null; 138 connect-timeout = 1; 139 }; 140 }; 141 142 initiatorRootDisk = 143 { 144 config, 145 pkgs, 146 modulesPath, 147 lib, 148 ... 149 }: 150 { 151 boot.initrd.network.enable = true; 152 boot.loader.grub.enable = false; 153 154 boot.kernelParams = lib.mkOverride 5 ([ 155 "boot.shell_on_fail" 156 "console=tty1" 157 "ip=192.168.1.1:::255.255.255.0::ens9:none" 158 "ip=192.168.2.1:::255.255.255.0::ens10:none" 159 ]); 160 161 # defaults to true, puts some code in the initrd that tries to mount an overlayfs on /nix/store 162 virtualisation.writableStore = false; 163 virtualisation.vlans = [ 164 1 165 2 166 ]; 167 168 services.multipath = { 169 enable = true; 170 defaults = '' 171 find_multipaths yes 172 user_friendly_names yes 173 ''; 174 pathGroups = [ 175 { 176 alias = 123456; 177 wwid = "3600140592b17c3f6b404168b082ceeb7"; 178 } 179 ]; 180 }; 181 182 fileSystems = lib.mkOverride 5 { 183 "/" = { 184 fsType = "xfs"; 185 device = "/dev/mapper/123456"; 186 options = [ "_netdev" ]; 187 }; 188 }; 189 190 boot.initrd.extraFiles."etc/multipath/wwids".source = 191 pkgs.writeText "wwids" "/3600140592b17c3f6b404168b082ceeb7/"; 192 193 boot.iscsi-initiator = { 194 discoverPortal = "target"; 195 name = initiatorName; 196 target = targetName; 197 extraIscsiCommands = '' 198 iscsiadm -m discovery -o update -t sendtargets -p 192.168.2.3 --login 199 ''; 200 }; 201 }; 202 203 }; 204 205 testScript = 206 { nodes, ... }: 207 '' 208 target.start() 209 target.wait_for_unit("iscsi-target.service") 210 211 initiatorAuto.start() 212 213 initiatorAuto.wait_for_unit("iscsid.service") 214 initiatorAuto.wait_for_unit("iscsi.service") 215 initiatorAuto.get_unit_info("iscsi") 216 217 # Expecting this to fail since we should already know about 192.168.1.3 218 initiatorAuto.fail("iscsiadm -m discovery -o update -t sendtargets -p 192.168.1.3 --login") 219 # Expecting this to succeed since we don't yet know about 192.168.2.3 220 initiatorAuto.succeed("iscsiadm -m discovery -o update -t sendtargets -p 192.168.2.3 --login") 221 222 # /dev/sda is provided by iscsi on target 223 initiatorAuto.succeed("set -x; while ! test -e /dev/sda; do sleep 1; done") 224 225 initiatorAuto.succeed("mkfs.xfs /dev/sda") 226 initiatorAuto.succeed("mkdir /mnt") 227 228 # Start by verifying /dev/sda and /dev/sdb are both the same disk 229 initiatorAuto.succeed("mount /dev/sda /mnt") 230 initiatorAuto.succeed("touch /mnt/hi") 231 initiatorAuto.succeed("umount /mnt") 232 233 initiatorAuto.succeed("mount /dev/sdb /mnt") 234 initiatorAuto.succeed("test -e /mnt/hi") 235 initiatorAuto.succeed("umount /mnt") 236 237 initiatorAuto.succeed("systemctl restart multipathd") 238 initiatorAuto.succeed("systemd-cat multipath -ll") 239 240 # Install our RootDisk machine to 123456, the alias to the device that multipath is now managing 241 initiatorAuto.succeed("mount /dev/mapper/123456 /mnt") 242 initiatorAuto.succeed("mkdir -p /mnt/etc/{multipath,iscsi}") 243 initiatorAuto.succeed("cp -r /etc/multipath/wwids /mnt/etc/multipath/wwids") 244 initiatorAuto.succeed("cp -r /etc/iscsi/{nodes,send_targets} /mnt/etc/iscsi") 245 initiatorAuto.succeed( 246 "nixos-install --no-bootloader --no-root-passwd --system /etc/initiator-root-disk-closure" 247 ) 248 initiatorAuto.succeed("umount /mnt") 249 initiatorAuto.shutdown() 250 251 initiatorRootDisk.start() 252 initiatorRootDisk.wait_for_unit("multi-user.target") 253 initiatorRootDisk.wait_for_unit("iscsid") 254 255 # Log in over both nodes 256 initiatorRootDisk.fail("iscsiadm -m discovery -o update -t sendtargets -p 192.168.1.3 --login") 257 initiatorRootDisk.fail("iscsiadm -m discovery -o update -t sendtargets -p 192.168.2.3 --login") 258 initiatorRootDisk.succeed("systemctl restart multipathd") 259 initiatorRootDisk.succeed("systemd-cat multipath -ll") 260 261 # Verify we can write and sync the root disk 262 initiatorRootDisk.succeed("mkdir /scratch") 263 initiatorRootDisk.succeed("touch /scratch/both-up") 264 initiatorRootDisk.succeed("sync /scratch") 265 266 # Verify we can write to the root with ens9 (sda, 192.168.1.3) down 267 initiatorRootDisk.succeed("ip link set ens9 down") 268 initiatorRootDisk.succeed("touch /scratch/ens9-down") 269 initiatorRootDisk.succeed("sync /scratch") 270 initiatorRootDisk.succeed("ip link set ens9 up") 271 272 # todo: better way to wait until multipath notices the link is back 273 initiatorRootDisk.succeed("sleep 5") 274 initiatorRootDisk.succeed("touch /scratch/both-down") 275 initiatorRootDisk.succeed("sync /scratch") 276 277 # Verify we can write to the root with ens10 (sdb, 192.168.2.3) down 278 initiatorRootDisk.succeed("ip link set ens10 down") 279 initiatorRootDisk.succeed("touch /scratch/ens10-down") 280 initiatorRootDisk.succeed("sync /scratch") 281 initiatorRootDisk.succeed("ip link set ens10 up") 282 initiatorRootDisk.succeed("touch /scratch/ens10-down") 283 initiatorRootDisk.succeed("sync /scratch") 284 285 initiatorRootDisk.succeed("ip link set ens9 up") 286 initiatorRootDisk.succeed("ip link set ens10 up") 287 initiatorRootDisk.shutdown() 288 289 # Verify we can boot with the target's eth1 down, forcing 290 # it to multipath via the second link 291 target.succeed("ip link set eth1 down") 292 initiatorRootDisk.start() 293 initiatorRootDisk.wait_for_unit("multi-user.target") 294 initiatorRootDisk.wait_for_unit("iscsid") 295 initiatorRootDisk.succeed("test -e /scratch/both-up") 296 ''; 297 } 298)