{ lib, nodes, ... }: let inherit (lib) attrNames concatMap concatMapStrings flip forEach head listToAttrs mkDefault mkOption nameValuePair optionalString range toLower types zipListsWith zipLists ; nodeNumbers = listToAttrs (zipListsWith nameValuePair (attrNames nodes) (range 1 254)); networkModule = { config, nodes, pkgs, ... }: let qemu-common = import ../qemu-common.nix { inherit lib pkgs; }; # Convert legacy VLANs to named interfaces and merge with explicit interfaces. vlansNumbered = forEach (zipLists config.virtualisation.vlans (range 1 255)) (v: { name = "eth${toString v.snd}"; vlan = v.fst; assignIP = true; }); explicitInterfaces = lib.mapAttrsToList (n: v: v // { name = n; }) config.virtualisation.interfaces; interfaces = vlansNumbered ++ explicitInterfaces; interfacesNumbered = zipLists interfaces (range 1 255); # Automatically assign IP addresses to requested interfaces. assignIPs = lib.filter (i: i.assignIP) interfaces; ipInterfaces = forEach assignIPs ( i: nameValuePair i.name { ipv4.addresses = [ { address = "192.168.${toString i.vlan}.${toString config.virtualisation.test.nodeNumber}"; prefixLength = 24; } ]; ipv6.addresses = [ { address = "2001:db8:${toString i.vlan}::${toString config.virtualisation.test.nodeNumber}"; prefixLength = 64; } ]; } ); qemuOptions = lib.flatten ( forEach interfacesNumbered ( { fst, snd }: qemu-common.qemuNICFlags snd fst.vlan config.virtualisation.test.nodeNumber ) ); udevRules = forEach interfacesNumbered ( { fst, snd }: # MAC Addresses for QEMU network devices are lowercase, and udev string comparison is case-sensitive. ''SUBSYSTEM=="net",ACTION=="add",ATTR{address}=="${toLower (qemu-common.qemuNicMac fst.vlan config.virtualisation.test.nodeNumber)}",NAME="${fst.name}"'' ); networkConfig = { networking.hostName = mkDefault config.virtualisation.test.nodeName; networking.interfaces = listToAttrs ipInterfaces; networking.primaryIPAddress = optionalString (ipInterfaces != [ ]) (head (head ipInterfaces).value.ipv4.addresses).address; networking.primaryIPv6Address = optionalString (ipInterfaces != [ ]) (head (head ipInterfaces).value.ipv6.addresses).address; # Put the IP addresses of all VMs in this machine's # /etc/hosts file. If a machine has multiple # interfaces, use the IP address corresponding to # the first interface (i.e. the first network in its # virtualisation.vlans option). networking.extraHosts = flip concatMapStrings (attrNames nodes) ( m': let config = nodes.${m'}; hostnames = optionalString ( config.networking.domain != null ) "${config.networking.hostName}.${config.networking.domain} " + "${config.networking.hostName}\n"; in optionalString ( config.networking.primaryIPAddress != "" ) "${config.networking.primaryIPAddress} ${hostnames}" + optionalString (config.networking.primaryIPv6Address != "") ( "${config.networking.primaryIPv6Address} ${hostnames}" ) ); virtualisation.qemu.options = qemuOptions; boot.initrd.services.udev.rules = concatMapStrings (x: x + "\n") udevRules; }; in { key = "network-interfaces"; config = networkConfig // { # Expose the networkConfig items for tests like nixops # that need to recreate the network config. system.build.networkConfig = networkConfig; }; }; nodeNumberModule = ( regular@{ config, name, ... }: { options = { virtualisation.test.nodeName = mkOption { internal = true; default = name; # We need to force this in specilisations, otherwise it'd be # readOnly = true; description = '' The `name` in `nodes.`; stable across `specialisations`. ''; }; virtualisation.test.nodeNumber = mkOption { internal = true; type = types.int; readOnly = true; default = nodeNumbers.${config.virtualisation.test.nodeName}; description = '' A unique number assigned for each node in `nodes`. ''; }; # specialisations override the `name` module argument, # so we push the real `virtualisation.test.nodeName`. specialisation = mkOption { type = types.attrsOf ( types.submodule { options.configuration = mkOption { type = types.submoduleWith { modules = [ { config.virtualisation.test.nodeName = # assert regular.config.virtualisation.test.nodeName != "configuration"; regular.config.virtualisation.test.nodeName; } ]; }; }; } ); }; }; } ); in { config = { extraBaseModules = { imports = [ networkModule nodeNumberModule ]; }; }; }