summary refs log tree commit diff
path: root/nixos/tests/xmonad.nix
blob: a2fb38e53bd15d622b355faae86247a05e223370 (plain) (blame)
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
import ./make-test-python.nix ({ pkgs, ...}:

let
  mkConfig = name: keys: ''
    import XMonad
    import XMonad.Operations (restart)
    import XMonad.Util.EZConfig
    import XMonad.Util.SessionStart
    import Control.Monad (when)
    import Text.Printf (printf)
    import System.Posix.Process (executeFile)
    import System.Info (arch,os)
    import System.Environment (getArgs)
    import System.FilePath ((</>))

    main = launch $ def { startupHook = startup } `additionalKeysP` myKeys

    startup = isSessionStart >>= \sessInit ->
      spawn "touch /tmp/${name}"
        >> if sessInit then setSessionStarted else spawn "xterm"

    myKeys = [${builtins.concatStringsSep ", " keys}]

    compiledConfig = printf "xmonad-%s-%s" arch os

    compileRestart resume =
      whenX (recompile True) $
        when resume writeStateToFile
          *> catchIO
            ( do
                dir <- getXMonadDataDir
                args <- getArgs
                executeFile (dir </> compiledConfig) False args Nothing
            )
  '';

  oldKeys =
    [ ''("M-C-x", spawn "xterm")''
      ''("M-q", restart "xmonad" True)''
      ''("M-C-q", compileRestart True)''
      ''("M-C-t", spawn "touch /tmp/somefile")'' # create somefile
    ];

  newKeys =
    [ ''("M-C-x", spawn "xterm")''
      ''("M-q", restart "xmonad" True)''
      ''("M-C-q", compileRestart True)''
      ''("M-C-r", spawn "rm /tmp/somefile")'' # delete somefile
    ];

  newConfig = pkgs.writeText "xmonad.hs" (mkConfig "newXMonad" newKeys);
in {
  name = "xmonad";
  meta = with pkgs.lib.maintainers; {
    maintainers = [ nequissimus ivanbrennan ];
  };

  machine = { pkgs, ... }: {
    imports = [ ./common/x11.nix ./common/user-account.nix ];
    test-support.displayManager.auto.user = "alice";
    services.xserver.displayManager.defaultSession = "none+xmonad";
    services.xserver.windowManager.xmonad = {
      enable = true;
      enableConfiguredRecompile = true;
      enableContribAndExtras = true;
      extraPackages = with pkgs.haskellPackages; haskellPackages: [ xmobar ];
      config = mkConfig "oldXMonad" oldKeys;
    };
  };

  testScript = { nodes, ... }: let
    user = nodes.machine.config.users.users.alice;
  in ''
    machine.wait_for_x()
    machine.wait_for_file("${user.home}/.Xauthority")
    machine.succeed("xauth merge ${user.home}/.Xauthority")
    machine.send_key("alt-ctrl-x")
    machine.wait_for_window("${user.name}.*machine")
    machine.sleep(1)
    machine.screenshot("terminal1")
    machine.succeed("rm /tmp/oldXMonad")
    machine.send_key("alt-q")
    machine.wait_for_file("/tmp/oldXMonad")
    machine.wait_for_window("${user.name}.*machine")
    machine.sleep(1)
    machine.screenshot("terminal2")

    # /tmp/somefile should not exist yet
    machine.fail("stat /tmp/somefile")

    # original config has a keybinding that creates somefile
    machine.send_key("alt-ctrl-t")
    machine.wait_for_file("/tmp/somefile")

    # set up the new config
    machine.succeed("mkdir -p ${user.home}/.xmonad")
    machine.copy_from_host("${newConfig}", "${user.home}/.xmonad/xmonad.hs")

    # recompile xmonad using the new config
    machine.send_key("alt-ctrl-q")
    machine.wait_for_file("/tmp/newXMonad")

    # new config has a keybinding that deletes somefile
    machine.send_key("alt-ctrl-r")
    machine.wait_until_fails("stat /tmp/somefile", timeout=30)

    # restart with the old config, and confirm the old keybinding is back
    machine.succeed("rm /tmp/oldXMonad")
    machine.send_key("alt-q")
    machine.wait_for_file("/tmp/oldXMonad")
    machine.send_key("alt-ctrl-t")
    machine.wait_for_file("/tmp/somefile")
  '';
})