{ system, minimal ? false }: with import ./build-vms.nix { inherit system minimal; }; with pkgs; rec { inherit pkgs; testDriver = stdenv.mkDerivation { name = "nixos-test-driver"; buildInputs = [ makeWrapper perl ]; unpackPhase = "true"; installPhase = '' mkdir -p $out/bin cp ${./test-driver/test-driver.pl} $out/bin/nixos-test-driver chmod u+x $out/bin/nixos-test-driver libDir=$out/lib/perl5/site_perl mkdir -p $libDir cp ${./test-driver/Machine.pm} $libDir/Machine.pm cp ${./test-driver/Logger.pm} $libDir/Logger.pm wrapProgram $out/bin/nixos-test-driver \ --prefix PATH : "${pkgs.qemu_kvm}/bin:${pkgs.vde2}/bin:${imagemagick}/bin:${coreutils}/bin" \ --prefix PERL5LIB : "${lib.makePerlPath [ perlPackages.TermReadLineGnu perlPackages.XMLWriter perlPackages.IOTty ]}:$out/lib/perl5/site_perl" ''; }; # Run an automated test suite in the given virtual network. # `driver' is the script that runs the network. runTests = driver: stdenv.mkDerivation { name = "vm-test-run"; requiredSystemFeatures = [ "kvm" "nixos-test" ]; buildInputs = [ pkgs.libxslt ]; buildCommand = '' mkdir -p $out/nix-support LOGFILE=$out/log.xml tests='eval $ENV{testScript}; die $@ if $@;' ${driver}/bin/nixos-test-driver || failed=1 # Generate a pretty-printed log. xsltproc --output $out/log.html ${./test-driver/log2html.xsl} $out/log.xml ln -s ${./test-driver/logfile.css} $out/logfile.css ln -s ${./test-driver/treebits.js} $out/treebits.js touch $out/nix-support/hydra-build-products echo "report testlog $out log.html" >> $out/nix-support/hydra-build-products for i in */xchg/coverage-data; do mkdir -p $out/coverage-data mv $i $out/coverage-data/$(dirname $(dirname $i)) done [ -z "$failed" ] || touch $out/nix-support/failed ''; # */ }; # Generate a coverage report from the coverage data produced by # runTests. makeReport = x: runCommand "report" { buildInputs = [rsync]; } '' mkdir -p $TMPDIR/gcov/ for d in ${x}/coverage-data/*; do echo "doing $d" [ -n "$(ls -A "$d")" ] || continue for i in $(cd $d/nix/store && ls); do if ! test -e $TMPDIR/gcov/nix/store/$i; then echo "copying $i" mkdir -p $TMPDIR/gcov/$(echo $i | cut -c34-) rsync -rv /nix/store/$i/.build/* $TMPDIR/gcov/ fi done chmod -R u+w $TMPDIR/gcov find $TMPDIR/gcov -name "*.gcda" -exec rm {} \; for i in $(cd $d/nix/store && ls); do rsync -rv $d/nix/store/$i/.build/* $TMPDIR/gcov/ done find $TMPDIR/gcov -name "*.gcda" -exec chmod 644 {} \; echo "producing info..." ${pkgs.lcov}/bin/geninfo --ignore-errors source,gcov $TMPDIR/gcov --output-file $TMPDIR/app.info cat $TMPDIR/app.info >> $TMPDIR/full.info done echo "making report..." mkdir -p $out/coverage ${pkgs.lcov}/bin/genhtml --show-details $TMPDIR/full.info -o $out/coverage cp $TMPDIR/full.info $out/coverage/ mkdir -p $out/nix-support cat ${x}/nix-support/hydra-build-products >> $out/nix-support/hydra-build-products echo "report coverage $out/coverage" >> $out/nix-support/hydra-build-products [ ! -e ${x}/nix-support/failed ] || touch $out/nix-support/failed ''; # */ makeTest = testFun: complete (call testFun); makeTests = testsFun: lib.mapAttrs (name: complete) (call testsFun); apply = makeTest; # compatibility call = f: f { inherit pkgs system; }; complete = t: t // rec { nodes = buildVirtualNetwork ( if t ? nodes then t.nodes else if t ? machine then { machine = t.machine; } else { } ); testScript = # Call the test script with the computed nodes. if builtins.isFunction t.testScript then t.testScript { inherit nodes; } else t.testScript; vlans = map (m: m.config.virtualisation.vlans) (lib.attrValues nodes); vms = map (m: m.config.system.build.vm) (lib.attrValues nodes); # Generate onvenience wrappers for running the test driver # interactively with the specified network, and for starting the # VMs from the command line. driver = runCommand "nixos-test-driver" { buildInputs = [ makeWrapper]; inherit testScript; preferLocalBuild = true; } '' mkdir -p $out/bin echo "$testScript" > $out/test-script ln -s ${testDriver}/bin/nixos-test-driver $out/bin/ vms="$(for i in ${toString vms}; do echo $i/bin/run-*-vm; done)" wrapProgram $out/bin/nixos-test-driver \ --add-flags "$vms" \ --run "testScript=\"\$(cat $out/test-script)\"" \ --set testScript '"$testScript"' \ --set VLANS '"${toString vlans}"' ln -s ${testDriver}/bin/nixos-test-driver $out/bin/nixos-run-vms wrapProgram $out/bin/nixos-run-vms \ --add-flags "$vms" \ --set tests '"startAll; joinAll;"' \ --set VLANS '"${toString vlans}"' \ ${lib.optionalString (builtins.length vms == 1) "--set USE_SERIAL 1"} ''; # " test = runTests driver; report = makeReport test; }; runInMachine = { drv , machine , preBuild ? "" , postBuild ? "" , ... # ??? }: let vm = buildVM { } [ machine { key = "hostname"; networking.hostName = "client"; } ]; buildrunner = writeText "vm-build" '' source $1 ${coreutils}/bin/mkdir -p $TMPDIR cd $TMPDIR $origBuilder $origArgs exit $? ''; testscript = '' startAll; ${preBuild} $client->succeed("env -i ${pkgs.bash}/bin/bash ${buildrunner} /tmp/xchg/saved-env >&2"); ${postBuild} ''; vmRunCommand = writeText "vm-run" '' ${coreutils}/bin/mkdir $out ${coreutils}/bin/mkdir -p vm-state-client/xchg export > vm-state-client/xchg/saved-env export tests='${testscript}' ${testDriver}/bin/nixos-test-driver ${vm.config.system.build.vm}/bin/run-*-vm ''; # */ in lib.overrideDerivation drv (attrs: { requiredSystemFeatures = [ "kvm" ]; builder = "${bash}/bin/sh"; args = ["-e" vmRunCommand]; origArgs = attrs.args; origBuilder = attrs.builder; }); runInMachineWithX = { require ? [], ... } @ args: let client = { config, pkgs, ... }: { inherit require; virtualisation.memorySize = 1024; services.xserver.enable = true; services.xserver.displayManager.slim.enable = false; services.xserver.displayManager.auto.enable = true; services.xserver.windowManager.default = "icewm"; services.xserver.windowManager.icewm.enable = true; services.xserver.desktopManager.default = "none"; }; in runInMachine ({ machine = client; preBuild = '' $client->waitForX; ''; } // args); simpleTest = as: (makeTest ({ ... }: as)).test; }