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