f9611764c6
In my earlier commit manual: Don't suggest exposing VM port to local network. I made a side change titled Use `127.0.0.1` also on the VM side, otherwise connections to services that, in the VM, bind to `127.0.0.1` only (doing the safe approach) do not work. Unfortunately, that was wrong: QEMU inside the VM always communicates via the virtualised Ethernet interface, not via the VM's loopback interface. So trying to connect to `127.0.0.1` on the VM's side cannot work.
104 lines
3.5 KiB
Markdown
104 lines
3.5 KiB
Markdown
# Running Tests interactively {#sec-running-nixos-tests-interactively}
|
|
|
|
The test itself can be run interactively. This is particularly useful
|
|
when developing or debugging a test:
|
|
|
|
```ShellSession
|
|
$ nix-build . -A nixosTests.login.driverInteractive
|
|
$ ./result/bin/nixos-test-driver
|
|
[...]
|
|
>>>
|
|
```
|
|
|
|
You can then take any Python statement, e.g.
|
|
|
|
```py
|
|
>>> start_all()
|
|
>>> test_script()
|
|
>>> machine.succeed("touch /tmp/foo")
|
|
>>> print(machine.succeed("pwd")) # Show stdout of command
|
|
```
|
|
|
|
The function `test_script` executes the entire test script and drops you
|
|
back into the test driver command line upon its completion. This allows
|
|
you to inspect the state of the VMs after the test (e.g. to debug the
|
|
test script).
|
|
|
|
## Shell access in interactive mode {#sec-nixos-test-shell-access}
|
|
|
|
The function `<yourmachine>.shell_interact()` grants access to a shell running
|
|
inside a virtual machine. To use it, replace `<yourmachine>` with the name of a
|
|
virtual machine defined in the test, for example: `machine.shell_interact()`.
|
|
Keep in mind that this shell may not display everything correctly as it is
|
|
running within an interactive Python REPL, and logging output from the virtual
|
|
machine may overwrite input and output from the guest shell:
|
|
|
|
```py
|
|
>>> machine.shell_interact()
|
|
machine: Terminal is ready (there is no initial prompt):
|
|
$ hostname
|
|
machine
|
|
```
|
|
|
|
As an alternative, you can proxy the guest shell to a local TCP server by first
|
|
starting a TCP server in a terminal using the command:
|
|
|
|
```ShellSession
|
|
$ socat 'READLINE,PROMPT=$ ' tcp-listen:4444,reuseaddr`
|
|
```
|
|
|
|
In the terminal where the test driver is running, connect to this server by
|
|
using:
|
|
|
|
```py
|
|
>>> machine.shell_interact("tcp:127.0.0.1:4444")
|
|
```
|
|
|
|
Once the connection is established, you can enter commands in the socat terminal
|
|
where socat is running.
|
|
|
|
## Port forwarding to NixOS test VMs {#sec-nixos-test-port-forwarding}
|
|
|
|
If your test has only a single VM, you may use e.g.
|
|
|
|
```ShellSession
|
|
$ QEMU_NET_OPTS="hostfwd=tcp:127.0.0.1:2222-:22" ./result/bin/nixos-test-driver
|
|
```
|
|
|
|
to port-forward a port in the VM (here `22`) to the host machine (here port `2222`).
|
|
|
|
This naturally does not work when multiple machines are involved,
|
|
since a single port on the host cannot forward to multiple VMs.
|
|
|
|
If the test defines multiple machines, you may opt to _temporarily_ set
|
|
`virtualisation.forwardPorts` in the test definition for debugging.
|
|
|
|
Such port forwardings connect via the VM's virtual network interface.
|
|
Thus they cannot connect to ports that are only bound to the VM's
|
|
loopback interface (`127.0.0.1`), and the VM's NixOS firewall
|
|
must be configured to allow these connections.
|
|
|
|
## Reuse VM state {#sec-nixos-test-reuse-vm-state}
|
|
|
|
You can re-use the VM states coming from a previous run by setting the
|
|
`--keep-vm-state` flag.
|
|
|
|
```ShellSession
|
|
$ ./result/bin/nixos-test-driver --keep-vm-state
|
|
```
|
|
|
|
The machine state is stored in the `$TMPDIR/vm-state-machinename`
|
|
directory.
|
|
|
|
## Interactive-only test configuration {#sec-nixos-test-interactive-configuration}
|
|
|
|
The `.driverInteractive` attribute combines the regular test configuration with
|
|
definitions from the [`interactive` submodule](#test-opt-interactive). This gives you
|
|
a more usable, graphical, but slightly different configuration.
|
|
|
|
You can add your own interactive-only test configuration by adding extra
|
|
configuration to the [`interactive` submodule](#test-opt-interactive).
|
|
|
|
To interactively run only the regular configuration, build the `<test>.driver` attribute
|
|
instead, and call it with the flag `result/bin/nixos-test-driver --interactive`.
|