"""Tests for hytop.core.ssh.collect_from_host (subprocess mocked).""" from __future__ import annotations import subprocess from unittest.mock import MagicMock, patch from hytop.core.ssh import collect_from_host def _make_proc(returncode=0, stdout="", stderr=""): m = MagicMock() m.returncode = returncode m.stdout = stdout m.stderr = stderr return m class TestCollectFromHostLocal: @patch("hytop.core.ssh.subprocess.run") def test_success_returns_no_error(self, mock_run): mock_run.return_value = _make_proc(stdout='{"card0":{}}') result = collect_from_host( "localhost", ssh_timeout=5, cmd_timeout=10, hy_smi_args=["--json"] ) assert result.error is None assert result.host == "localhost" @patch("hytop.core.ssh.subprocess.run") def test_local_invokes_hy_smi_directly(self, mock_run): mock_run.return_value = _make_proc() collect_from_host("localhost", ssh_timeout=5, cmd_timeout=10, hy_smi_args=["--json"]) cmd = mock_run.call_args[0][0] assert cmd[0] == "hy-smi" assert "ssh" not in cmd @patch("hytop.core.ssh.subprocess.run") def test_127_0_0_1_treated_as_local(self, mock_run): mock_run.return_value = _make_proc() collect_from_host("127.0.0.1", ssh_timeout=5, cmd_timeout=10, hy_smi_args=["--json"]) cmd = mock_run.call_args[0][0] assert cmd[0] == "hy-smi" @patch("hytop.core.ssh.subprocess.run") def test_nonzero_exit_returns_error(self, mock_run): mock_run.return_value = _make_proc(returncode=1, stderr="permission denied") result = collect_from_host( "localhost", ssh_timeout=5, cmd_timeout=10, hy_smi_args=["--json"] ) assert result.error is not None assert "exit 1" in result.error @patch( "hytop.core.ssh.subprocess.run", side_effect=subprocess.TimeoutExpired("cmd", 10), ) def test_timeout_returns_error(self, mock_run): result = collect_from_host( "localhost", ssh_timeout=5, cmd_timeout=10, hy_smi_args=["--json"] ) assert result.error is not None assert "timeout" in result.error @patch("hytop.core.ssh.subprocess.run", side_effect=OSError("no such file")) def test_oserror_returns_error(self, mock_run): result = collect_from_host( "localhost", ssh_timeout=5, cmd_timeout=10, hy_smi_args=["--json"] ) assert result.error is not None assert "no such file" in result.error class TestCollectFromHostRemote: @patch("hytop.core.ssh.subprocess.run") def test_remote_uses_ssh(self, mock_run): mock_run.return_value = _make_proc(stdout="{}") collect_from_host("node01", ssh_timeout=5, cmd_timeout=10, hy_smi_args=["--json"]) cmd = mock_run.call_args[0][0] assert cmd[0] == "ssh" @patch("hytop.core.ssh.subprocess.run") def test_remote_hostname_in_cmd(self, mock_run): mock_run.return_value = _make_proc(stdout="{}") collect_from_host("node01", ssh_timeout=5, cmd_timeout=10, hy_smi_args=["--json"]) cmd = mock_run.call_args[0][0] assert "node01" in cmd @patch("hytop.core.ssh.subprocess.run") def test_remote_batch_mode_set(self, mock_run): mock_run.return_value = _make_proc(stdout="{}") collect_from_host("node01", ssh_timeout=5, cmd_timeout=10, hy_smi_args=["--json"]) cmd = mock_run.call_args[0][0] assert "BatchMode=yes" in cmd @patch("hytop.core.ssh.subprocess.run") def test_hy_smi_args_forwarded(self, mock_run): mock_run.return_value = _make_proc(stdout="{}") collect_from_host( "node01", ssh_timeout=5, cmd_timeout=10, hy_smi_args=["--json", "--showtemp"], ) cmd = mock_run.call_args[0][0] assert "--json" in cmd assert "--showtemp" in cmd