at 25.11-pre 8.7 kB view raw
1import ../make-test-python.nix ( 2 { pkgs, lib, ... }: 3 let 4 quadletContainerFile = pkgs.writeText "quadlet.container" '' 5 [Unit] 6 Description=A test quadlet container 7 8 [Container] 9 Image=localhost/scratchimg:latest 10 Exec=bash -c 'trap exit SIGTERM SIGINT; while true; do sleep 1; done' 11 ContainerName=quadlet 12 Volume=/nix/store:/nix/store 13 Volume=/run/current-system/sw/bin:/bin 14 15 [Install] 16 WantedBy=default.target 17 ''; 18 in 19 { 20 name = "podman"; 21 meta = { 22 maintainers = lib.teams.podman.members; 23 }; 24 25 nodes = { 26 rootful = 27 { pkgs, ... }: 28 { 29 virtualisation.podman.enable = true; 30 31 # hack to ensure that podman built with and without zfs in extraPackages is cached 32 boot.supportedFilesystems = [ "zfs" ]; 33 networking.hostId = "00000000"; 34 }; 35 rootless = 36 { pkgs, ... }: 37 { 38 virtualisation.podman.enable = true; 39 40 users.users.alice = { 41 isNormalUser = true; 42 }; 43 }; 44 dns = 45 { pkgs, ... }: 46 { 47 virtualisation.podman.enable = true; 48 49 virtualisation.podman.defaultNetwork.settings.dns_enabled = true; 50 }; 51 docker = 52 { pkgs, ... }: 53 { 54 virtualisation.podman.enable = true; 55 56 virtualisation.podman.dockerSocket.enable = true; 57 58 environment.systemPackages = [ 59 pkgs.docker-client 60 ]; 61 62 users.users.alice = { 63 isNormalUser = true; 64 extraGroups = [ "podman" ]; 65 }; 66 67 users.users.mallory = { 68 isNormalUser = true; 69 }; 70 }; 71 }; 72 73 testScript = '' 74 import shlex 75 76 77 def su_cmd(cmd, user = "alice"): 78 cmd = shlex.quote(cmd) 79 return f"su {user} -l -c {cmd}" 80 81 82 rootful.wait_for_unit("sockets.target") 83 rootless.wait_for_unit("sockets.target") 84 dns.wait_for_unit("sockets.target") 85 docker.wait_for_unit("sockets.target") 86 start_all() 87 88 with subtest("Run container as root with runc"): 89 rootful.succeed("tar cv --files-from /dev/null | podman import - scratchimg") 90 rootful.succeed( 91 "podman run --runtime=runc -d --name=sleeping -v /nix/store:/nix/store -v /run/current-system/sw/bin:/bin scratchimg /bin/sleep 10" 92 ) 93 rootful.succeed("podman ps | grep sleeping") 94 rootful.succeed("podman stop sleeping") 95 rootful.succeed("podman rm sleeping") 96 97 with subtest("Run container as root with crun"): 98 rootful.succeed("tar cv --files-from /dev/null | podman import - scratchimg") 99 rootful.succeed( 100 "podman run --runtime=crun -d --name=sleeping -v /nix/store:/nix/store -v /run/current-system/sw/bin:/bin scratchimg /bin/sleep 10" 101 ) 102 rootful.succeed("podman ps | grep sleeping") 103 rootful.succeed("podman stop sleeping") 104 rootful.succeed("podman rm sleeping") 105 106 with subtest("Run container as root with the default backend"): 107 rootful.succeed("tar cv --files-from /dev/null | podman import - scratchimg") 108 rootful.succeed( 109 "podman run -d --name=sleeping -v /nix/store:/nix/store -v /run/current-system/sw/bin:/bin scratchimg /bin/sleep 10" 110 ) 111 rootful.succeed("podman ps | grep sleeping") 112 rootful.succeed("podman stop sleeping") 113 rootful.succeed("podman rm sleeping") 114 115 # start systemd session for rootless 116 rootless.succeed("loginctl enable-linger alice") 117 rootless.succeed(su_cmd("whoami")) 118 rootless.sleep(1) 119 120 with subtest("Run container rootless with runc"): 121 rootless.succeed(su_cmd("tar cv --files-from /dev/null | podman import - scratchimg")) 122 rootless.succeed( 123 su_cmd( 124 "podman run --runtime=runc -d --name=sleeping -v /nix/store:/nix/store -v /run/current-system/sw/bin:/bin scratchimg /bin/sleep 10" 125 ) 126 ) 127 rootless.succeed(su_cmd("podman ps | grep sleeping")) 128 rootless.succeed(su_cmd("podman stop sleeping")) 129 rootless.succeed(su_cmd("podman rm sleeping")) 130 131 with subtest("Run container rootless with crun"): 132 rootless.succeed(su_cmd("tar cv --files-from /dev/null | podman import - scratchimg")) 133 rootless.succeed( 134 su_cmd( 135 "podman run --runtime=crun -d --name=sleeping -v /nix/store:/nix/store -v /run/current-system/sw/bin:/bin scratchimg /bin/sleep 10" 136 ) 137 ) 138 rootless.succeed(su_cmd("podman ps | grep sleeping")) 139 rootless.succeed(su_cmd("podman stop sleeping")) 140 rootless.succeed(su_cmd("podman rm sleeping")) 141 142 with subtest("Run container rootless with the default backend"): 143 rootless.succeed(su_cmd("tar cv --files-from /dev/null | podman import - scratchimg")) 144 rootless.succeed( 145 su_cmd( 146 "podman run -d --name=sleeping -v /nix/store:/nix/store -v /run/current-system/sw/bin:/bin scratchimg /bin/sleep 10" 147 ) 148 ) 149 rootless.succeed(su_cmd("podman ps | grep sleeping")) 150 rootless.succeed(su_cmd("podman stop sleeping")) 151 rootless.succeed(su_cmd("podman rm sleeping")) 152 153 with subtest("rootlessport"): 154 rootless.succeed(su_cmd("tar cv --files-from /dev/null | podman import - scratchimg")) 155 rootless.succeed( 156 su_cmd( 157 "podman run -d -p 9000:8888 --name=rootlessport -v /nix/store:/nix/store -v /run/current-system/sw/bin:/bin -w ${pkgs.writeTextDir "index.html" "<h1>Testing</h1>"} scratchimg ${pkgs.python3}/bin/python -m http.server 8888" 158 ) 159 ) 160 rootless.succeed(su_cmd("podman ps | grep rootlessport")) 161 rootless.wait_until_succeeds(su_cmd("${pkgs.curl}/bin/curl localhost:9000 | grep Testing")) 162 rootless.succeed(su_cmd("podman stop rootlessport")) 163 rootless.succeed(su_cmd("podman rm rootlessport")) 164 165 with subtest("Run container with init"): 166 rootful.succeed( 167 "tar cv -C ${pkgs.pkgsStatic.busybox} . | podman import - busybox" 168 ) 169 pid = rootful.succeed("podman run --rm busybox readlink /proc/self").strip() 170 assert pid == "1" 171 pid = rootful.succeed("podman run --rm --init busybox readlink /proc/self").strip() 172 assert pid == "2" 173 174 with subtest("aardvark-dns"): 175 dns.succeed("tar cv --files-from /dev/null | podman import - scratchimg") 176 dns.succeed( 177 "podman run -d --name=webserver -v /nix/store:/nix/store -v /run/current-system/sw/bin:/bin -w ${pkgs.writeTextDir "index.html" "<h1>Testing</h1>"} scratchimg ${pkgs.python3}/bin/python -m http.server 8000" 178 ) 179 dns.succeed("podman ps | grep webserver") 180 dns.wait_until_succeeds( 181 "podman run --rm --name=client -v /nix/store:/nix/store -v /run/current-system/sw/bin:/bin scratchimg ${pkgs.curl}/bin/curl http://webserver:8000 | grep Testing" 182 ) 183 dns.succeed("podman stop webserver") 184 dns.succeed("podman rm webserver") 185 186 with subtest("A podman member can use the docker cli"): 187 docker.succeed(su_cmd("docker version")) 188 189 with subtest("Run container via docker cli"): 190 docker.succeed("tar cv --files-from /dev/null | podman import - scratchimg") 191 docker.succeed( 192 "docker run -d --name=sleeping -v /nix/store:/nix/store -v /run/current-system/sw/bin:/bin localhost/scratchimg /bin/sleep 10" 193 ) 194 docker.succeed("docker ps | grep sleeping") 195 docker.succeed("podman ps | grep sleeping") 196 docker.succeed("docker stop sleeping") 197 docker.succeed("docker rm sleeping") 198 199 with subtest("A podman non-member can not use the docker cli"): 200 docker.fail(su_cmd("docker version", user="mallory")) 201 202 with subtest("A rootless quadlet container service is created"): 203 dir = "/home/alice/.config/containers/systemd" 204 rootless.succeed(su_cmd("tar cv --files-from /dev/null | podman import - scratchimg")) 205 rootless.succeed(su_cmd(f"mkdir -p {dir}")) 206 rootless.succeed(su_cmd(f"cp -f ${quadletContainerFile} {dir}/quadlet.container")) 207 rootless.systemctl("daemon-reload", "alice") 208 rootless.systemctl("start quadlet", "alice") 209 rootless.wait_until_succeeds(su_cmd("podman ps | grep quadlet"), timeout=20) 210 rootless.systemctl("stop quadlet", "alice") 211 212 # TODO: add docker-compose test 213 214 ''; 215 } 216)