nixpkgs/nixos/tests/nixos-test-driver/console-log.nix
2025-09-12 16:46:03 +02:00

118 lines
5.2 KiB
Nix

# Run with: nix-build -A nixosTests.nixos-test-driver.console-log
{
name = "nixos-test-driver.console-log";
nodes = {
machine = {
# Configure the machine to print some distinctive messages to console
boot.kernelParams = [ "console=ttyS0,115200n8" ];
# TODO: add a system service that logs to /dev/console, and test that
# blocked on https://github.com/NixOS/nixpkgs/issues/442382
};
};
testScript = ''
# Start the machine and wait for it to boot
machine.start()
machine.wait_for_unit("multi-user.target")
with subtest("get_console_log returns console output"):
# Get the console log
console_log = machine.get_console_log()
# Verify it's a non-empty string
assert isinstance(console_log, str), f"Expected string, got {type(console_log)}"
assert len(console_log) > 0, "Console log should not be empty"
print(f"Console log length: {len(console_log)} characters")
print("Console log preview (first 500 chars):")
print(console_log[:500])
with subtest("console log contains boot messages"):
# Check for typical boot messages
assert "systemd" in console_log, "Console log should contain systemd messages"
# Test direct stderr capture
machine.succeed("echo 'DIRECT_TEST_MESSAGE_12345' >&2")
import time
time.sleep(1)
updated_log = machine.get_console_log()
assert "DIRECT_TEST_MESSAGE_12345" in updated_log, "Console log should capture stderr messages"
print("Successfully captured stderr message in console log")
with subtest("console log captures command output"):
# Execute a command that prints to console via kernel messages
machine.succeed("echo 'KERNEL_TEST_MESSAGE_12345' > /dev/kmsg")
# Wait a moment for the message to be captured
import time
time.sleep(2)
# Get updated console log
updated_log = machine.get_console_log()
if "KERNEL_TEST_MESSAGE_12345" in updated_log:
print("Successfully captured kernel message in console log")
else:
print("Kernel message not found in console log - this may be expected depending on log level")
with subtest("console log persists across multiple calls"):
# Get console log twice and verify they're consistent
log1 = machine.get_console_log()
log2 = machine.get_console_log()
# Both logs should contain the same content (log2 might have additional content)
assert "systemd" in log1, "First log should contain systemd messages"
assert "systemd" in log2, "Second log should contain systemd messages"
assert len(log2) >= len(log1), "Second log should be at least as long as first log"
assert "KERNEL_TEST_MESSAGE_12345" in log2, "Second log should contain our kernel test message"
with subtest("console log contains kernel messages"):
# Look for typical kernel boot messages
kernel_indicators = ["Linux version", "Command line:", "Kernel command line"]
kernel_found = any(indicator in console_log for indicator in kernel_indicators)
assert kernel_found, f"Console log should contain kernel messages. Indicators checked: {kernel_indicators}"
with subtest("console log is cleared on machine restart"):
# Get the current console log and verify it contains our test messages
pre_shutdown_log = machine.get_console_log()
assert "KERNEL_TEST_MESSAGE_12345" in pre_shutdown_log, "Pre-shutdown log should contain our kernel test message"
assert "DIRECT_TEST_MESSAGE_12345" in pre_shutdown_log, "Pre-shutdown log should contain our stderr test message"
pre_shutdown_length = len(pre_shutdown_log)
print(f"Pre-shutdown console log length: {pre_shutdown_length} characters")
# Shutdown the machine
machine.shutdown()
# Start the machine again
machine.start()
machine.wait_for_unit("multi-user.target")
# Get the console log after restart
post_restart_log = machine.get_console_log()
post_restart_length = len(post_restart_log)
print(f"Post-restart console log length: {post_restart_length} characters")
# Verify the old messages are gone (log should be fresh)
assert "KERNEL_TEST_MESSAGE_12345" not in post_restart_log, "Post-restart log should not contain old kernel test message"
assert "DIRECT_TEST_MESSAGE_12345" not in post_restart_log, "Post-restart log should not contain old stderr test message"
# Verify we still have boot messages (log should contain new boot sequence)
assert "systemd" in post_restart_log, "Post-restart log should contain systemd messages from new boot"
kernel_found_restart = any(indicator in post_restart_log for indicator in kernel_indicators)
assert kernel_found_restart, "Post-restart log should contain kernel messages from new boot"
# Add a new test message to verify the log is working after restart
machine.succeed("echo 'POST_RESTART_TEST_MESSAGE_67890' > /dev/kmsg")
import time
time.sleep(1)
final_log = machine.get_console_log()
assert "POST_RESTART_TEST_MESSAGE_67890" in final_log, "Should capture new messages after restart"
print("Console log successfully cleared on restart and is functional again")
print("All console log tests passed successfully!")
'';
}