mirror of
https://github.com/dw-0/kiauh.git
synced 2026-07-04 22:35:28 +05:00
feat(tests): add unit tests for shared utilities
This commit is contained in:
@@ -36,13 +36,15 @@ from utils.sys_utils import (
|
|||||||
update_system_package_lists,
|
update_system_package_lists,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
from kiauh import PROJECT_ROOT
|
||||||
|
|
||||||
|
|
||||||
def get_kiauh_version() -> str:
|
def get_kiauh_version() -> str:
|
||||||
"""
|
"""
|
||||||
Helper method to get the current KIAUH version by reading the latest tag
|
Helper method to get the current KIAUH version by reading the latest tag
|
||||||
:return: string of the latest tag or a default value if no tags exist
|
:return: string of the latest tag or a default value if no tags exist
|
||||||
"""
|
"""
|
||||||
tags: List[str] = get_local_tags(Path(__file__).parent.parent)
|
tags: List[str] = get_local_tags(PROJECT_ROOT)
|
||||||
if tags:
|
if tags:
|
||||||
return tags[-1]
|
return tags[-1]
|
||||||
else:
|
else:
|
||||||
|
|||||||
@@ -67,7 +67,7 @@ def git_pull_wrapper(target_dir: Path) -> None:
|
|||||||
Logger.print_status("Updating repository ...")
|
Logger.print_status("Updating repository ...")
|
||||||
try:
|
try:
|
||||||
git_cmd_pull(target_dir)
|
git_cmd_pull(target_dir)
|
||||||
except CalledProcessError:
|
except (CalledProcessError, GitException):
|
||||||
log = "An unexpected error occured during updating the repository."
|
log = "An unexpected error occured during updating the repository."
|
||||||
Logger.print_error(log)
|
Logger.print_error(log)
|
||||||
return
|
return
|
||||||
@@ -102,6 +102,9 @@ def get_current_branch(repo: Path) -> str | None:
|
|||||||
:param repo: Path to the local Git repository
|
:param repo: Path to the local Git repository
|
||||||
:return: Current branch or None if not determinable
|
:return: Current branch or None if not determinable
|
||||||
"""
|
"""
|
||||||
|
if not repo.exists() or not repo.joinpath(".git").exists():
|
||||||
|
return None
|
||||||
|
|
||||||
try:
|
try:
|
||||||
cmd = ["git", "branch", "--show-current"]
|
cmd = ["git", "branch", "--show-current"]
|
||||||
result: str = check_output(cmd, stderr=DEVNULL, cwd=repo).decode(
|
result: str = check_output(cmd, stderr=DEVNULL, cwd=repo).decode(
|
||||||
@@ -120,6 +123,8 @@ def get_local_tags(repo_path: Path, _filter: str | None = None) -> List[str]:
|
|||||||
:param _filter: Optional filter to filter the tags by
|
:param _filter: Optional filter to filter the tags by
|
||||||
:return: List of tags
|
:return: List of tags
|
||||||
"""
|
"""
|
||||||
|
if not repo_path.exists() or not repo_path.joinpath(".git").is_dir():
|
||||||
|
return []
|
||||||
|
|
||||||
def parse_version(version: str) -> tuple:
|
def parse_version(version: str) -> tuple:
|
||||||
# Remove 'v' prefix if present
|
# Remove 'v' prefix if present
|
||||||
@@ -337,6 +342,11 @@ def git_cmd_checkout(branch: str | None, target_dir: Path) -> None:
|
|||||||
if branch is None:
|
if branch is None:
|
||||||
return
|
return
|
||||||
|
|
||||||
|
if not target_dir.exists() or not target_dir.joinpath(".git").exists():
|
||||||
|
log = f"'{target_dir}' is not a valid git repository."
|
||||||
|
Logger.print_error(log)
|
||||||
|
raise GitException(log)
|
||||||
|
|
||||||
try:
|
try:
|
||||||
command = ["git", "checkout", f"{branch}"]
|
command = ["git", "checkout", f"{branch}"]
|
||||||
run(command, cwd=target_dir, check=True)
|
run(command, cwd=target_dir, check=True)
|
||||||
@@ -349,6 +359,11 @@ def git_cmd_checkout(branch: str | None, target_dir: Path) -> None:
|
|||||||
|
|
||||||
|
|
||||||
def git_cmd_pull(target_dir: Path) -> None:
|
def git_cmd_pull(target_dir: Path) -> None:
|
||||||
|
if not target_dir.exists() or not target_dir.joinpath(".git").exists():
|
||||||
|
log = f"'{target_dir}' is not a valid git repository."
|
||||||
|
Logger.print_error(log)
|
||||||
|
raise GitException(log)
|
||||||
|
|
||||||
try:
|
try:
|
||||||
command = ["git", "pull"]
|
command = ["git", "pull"]
|
||||||
run(command, cwd=target_dir, check=True)
|
run(command, cwd=target_dir, check=True)
|
||||||
@@ -359,6 +374,11 @@ def git_cmd_pull(target_dir: Path) -> None:
|
|||||||
|
|
||||||
|
|
||||||
def rollback_repository(repo_dir: Path, instance: Type[InstanceType]) -> None:
|
def rollback_repository(repo_dir: Path, instance: Type[InstanceType]) -> None:
|
||||||
|
if not repo_dir.exists() or not repo_dir.joinpath(".git").exists():
|
||||||
|
log = f"'{repo_dir}' is not a valid git repository."
|
||||||
|
Logger.print_error(log)
|
||||||
|
raise GitException(log)
|
||||||
|
|
||||||
q1 = "How many commits do you want to roll back"
|
q1 = "How many commits do you want to roll back"
|
||||||
amount = get_number_input(q1, 1, allow_go_back=True)
|
amount = get_number_input(q1, 1, allow_go_back=True)
|
||||||
|
|
||||||
@@ -394,7 +414,7 @@ def get_repo_url(repo_dir: Path) -> str | None:
|
|||||||
:param repo_dir: Path to the git repository
|
:param repo_dir: Path to the git repository
|
||||||
:return: URL of the remote repository or None if not found
|
:return: URL of the remote repository or None if not found
|
||||||
"""
|
"""
|
||||||
if not repo_dir.exists():
|
if not repo_dir.exists() or not repo_dir.joinpath(".git").exists():
|
||||||
return None
|
return None
|
||||||
|
|
||||||
try:
|
try:
|
||||||
|
|||||||
@@ -0,0 +1,21 @@
|
|||||||
|
import sys
|
||||||
|
from pathlib import Path
|
||||||
|
|
||||||
|
import pytest
|
||||||
|
|
||||||
|
PROJECT_ROOT = Path(__file__).resolve().parents[3]
|
||||||
|
if str(PROJECT_ROOT) not in sys.path:
|
||||||
|
sys.path.insert(0, str(PROJECT_ROOT))
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.fixture(autouse=True)
|
||||||
|
def silence_logger(monkeypatch: pytest.MonkeyPatch) -> None:
|
||||||
|
for name in (
|
||||||
|
"print_info",
|
||||||
|
"print_ok",
|
||||||
|
"print_warn",
|
||||||
|
"print_error",
|
||||||
|
"print_status",
|
||||||
|
"print_dialog",
|
||||||
|
):
|
||||||
|
monkeypatch.setattr(f"core.logger.Logger.{name}", lambda *a, **k: None)
|
||||||
@@ -0,0 +1,206 @@
|
|||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
from datetime import datetime
|
||||||
|
from pathlib import Path
|
||||||
|
from typing import List, Set
|
||||||
|
|
||||||
|
import pytest
|
||||||
|
from core.constants import GLOBAL_DEPS
|
||||||
|
from utils.common import (
|
||||||
|
check_install_dependencies,
|
||||||
|
convert_camelcase_to_kebabcase,
|
||||||
|
get_current_date,
|
||||||
|
get_install_status,
|
||||||
|
get_kiauh_version,
|
||||||
|
moonraker_exists,
|
||||||
|
trunc_string,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class TestGetKiauhVersion:
|
||||||
|
def test_uses_project_root(self, monkeypatch) -> None:
|
||||||
|
expected_root = Path(__file__).parent.parent.parent.parent
|
||||||
|
captured: List[Path] = []
|
||||||
|
|
||||||
|
def fake_get_local_tags(path: Path, _filter: str | None = None) -> List[str]:
|
||||||
|
captured.append(path)
|
||||||
|
return ["v6.3.0", "v6.3.1"]
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.common.get_local_tags", fake_get_local_tags)
|
||||||
|
result = get_kiauh_version()
|
||||||
|
|
||||||
|
assert captured == [expected_root]
|
||||||
|
assert result == "v6.3.1"
|
||||||
|
|
||||||
|
def test_fallback_when_no_tags(self, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr("utils.common.get_local_tags", lambda *_a, **_k: [])
|
||||||
|
assert get_kiauh_version() == "v?.?.?"
|
||||||
|
|
||||||
|
|
||||||
|
class TestConvertCamelcaseToKebabcase:
|
||||||
|
@pytest.mark.parametrize(
|
||||||
|
"name,expected",
|
||||||
|
[
|
||||||
|
("Klipper", "klipper"),
|
||||||
|
("Moonraker", "moonraker"),
|
||||||
|
("MoonrakerObico", "moonraker-obico"),
|
||||||
|
("HTTPResponse", "h-t-t-p-response"),
|
||||||
|
("already", "already"),
|
||||||
|
],
|
||||||
|
)
|
||||||
|
def test_converts(self, name: str, expected: str) -> None:
|
||||||
|
assert convert_camelcase_to_kebabcase(name) == expected
|
||||||
|
|
||||||
|
|
||||||
|
class TestGetCurrentDate:
|
||||||
|
def test_returns_formatted_values(self) -> None:
|
||||||
|
result = get_current_date()
|
||||||
|
now = datetime.today()
|
||||||
|
|
||||||
|
assert set(result.keys()) == {"date", "time"}
|
||||||
|
assert result["date"] == now.strftime("%Y%m%d")
|
||||||
|
assert result["time"] == now.strftime("%H%M%S")
|
||||||
|
|
||||||
|
|
||||||
|
class TestCheckInstallDependencies:
|
||||||
|
def test_with_global_and_custom(self, monkeypatch) -> None:
|
||||||
|
checked: Set[str] = set()
|
||||||
|
updated: List[bool] = []
|
||||||
|
installed_pkgs: List[List[str]] = []
|
||||||
|
|
||||||
|
def fake_check_package_install(deps: Set[str]) -> List[str]:
|
||||||
|
checked.update(deps)
|
||||||
|
return ["extra-pkg"]
|
||||||
|
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.common.check_package_install", fake_check_package_install
|
||||||
|
)
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.common.update_system_package_lists",
|
||||||
|
lambda silent: updated.append(silent),
|
||||||
|
)
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.common.install_system_packages",
|
||||||
|
lambda pkgs: installed_pkgs.append(pkgs),
|
||||||
|
)
|
||||||
|
|
||||||
|
check_install_dependencies({"custom-pkg"}, include_global=True)
|
||||||
|
|
||||||
|
assert "custom-pkg" in checked
|
||||||
|
assert all(dep in checked for dep in GLOBAL_DEPS)
|
||||||
|
assert updated == [False]
|
||||||
|
assert installed_pkgs == [["extra-pkg"]]
|
||||||
|
|
||||||
|
def test_no_requirements(self, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr("utils.common.check_package_install", lambda *_a, **_k: [])
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.common.update_system_package_lists",
|
||||||
|
lambda *a, **k: pytest.fail("should not update when nothing to install"),
|
||||||
|
)
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.common.install_system_packages",
|
||||||
|
lambda *a, **k: pytest.fail("should not install when nothing to install"),
|
||||||
|
)
|
||||||
|
|
||||||
|
check_install_dependencies({"pkg"})
|
||||||
|
|
||||||
|
|
||||||
|
class _FakeInstanceType:
|
||||||
|
def __init__(self, suffix: str):
|
||||||
|
self.suffix = suffix
|
||||||
|
|
||||||
|
def __eq__(self, other):
|
||||||
|
return isinstance(other, _FakeInstanceType) and self.suffix == other.suffix
|
||||||
|
|
||||||
|
|
||||||
|
class TestGetInstallStatus:
|
||||||
|
def test_not_installed(self, tmp_path: Path, monkeypatch) -> None:
|
||||||
|
repo = tmp_path / "repo"
|
||||||
|
env = tmp_path / "env"
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.common.get_current_branch", lambda *_a, **_k: None)
|
||||||
|
monkeypatch.setattr("utils.common.get_repo_name", lambda *_a, **_k: (None, None))
|
||||||
|
monkeypatch.setattr("utils.common.get_repo_url", lambda *_a, **_k: None)
|
||||||
|
monkeypatch.setattr("utils.common.get_local_commit", lambda *_a, **_k: None)
|
||||||
|
monkeypatch.setattr("utils.common.get_remote_commit", lambda *_a, **_k: None)
|
||||||
|
monkeypatch.setattr("utils.instance_utils.get_instances", lambda *_a, **_k: [])
|
||||||
|
|
||||||
|
status = get_install_status(repo, env, _FakeInstanceType)
|
||||||
|
|
||||||
|
assert status.status == 0
|
||||||
|
assert status.instances == 0
|
||||||
|
|
||||||
|
def test_fully_installed(self, tmp_path: Path, monkeypatch) -> None:
|
||||||
|
repo = tmp_path / "repo"
|
||||||
|
env = tmp_path / "env"
|
||||||
|
repo.mkdir()
|
||||||
|
env.mkdir()
|
||||||
|
(repo / ".git").mkdir()
|
||||||
|
extra_file = tmp_path / "extra"
|
||||||
|
extra_file.write_text("x")
|
||||||
|
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.instance_utils.get_instances", lambda *_a, **_k: [_FakeInstanceType("")]
|
||||||
|
)
|
||||||
|
monkeypatch.setattr("utils.common.get_current_branch", lambda *_a, **_k: "main")
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.common.get_repo_name", lambda *_a, **_k: ("dw-0", "kiauh")
|
||||||
|
)
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.common.get_repo_url", lambda *_a, **_k: "https://github.com/dw-0/kiauh"
|
||||||
|
)
|
||||||
|
monkeypatch.setattr("utils.common.get_local_commit", lambda *_a, **_k: "abc")
|
||||||
|
monkeypatch.setattr("utils.common.get_remote_commit", lambda *_a, **_k: "def")
|
||||||
|
|
||||||
|
status = get_install_status(repo, env, _FakeInstanceType, files=[extra_file])
|
||||||
|
|
||||||
|
assert status.status == 2
|
||||||
|
assert status.instances == 1
|
||||||
|
assert status.owner == "dw-0"
|
||||||
|
assert status.repo == "kiauh"
|
||||||
|
assert status.branch == "main"
|
||||||
|
assert status.local == "abc"
|
||||||
|
assert status.remote == "def"
|
||||||
|
|
||||||
|
def test_incomplete(self, tmp_path: Path, monkeypatch) -> None:
|
||||||
|
repo = tmp_path / "repo"
|
||||||
|
env = tmp_path / "env"
|
||||||
|
repo.mkdir()
|
||||||
|
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.instance_utils.get_instances", lambda *_a, **_k: [_FakeInstanceType("")]
|
||||||
|
)
|
||||||
|
monkeypatch.setattr("utils.common.get_current_branch", lambda *_a, **_k: "main")
|
||||||
|
monkeypatch.setattr("utils.common.get_repo_name", lambda *_a, **_k: (None, None))
|
||||||
|
monkeypatch.setattr("utils.common.get_repo_url", lambda *_a, **_k: None)
|
||||||
|
monkeypatch.setattr("utils.common.get_local_commit", lambda *_a, **_k: None)
|
||||||
|
monkeypatch.setattr("utils.common.get_remote_commit", lambda *_a, **_k: None)
|
||||||
|
|
||||||
|
status = get_install_status(repo, env, _FakeInstanceType)
|
||||||
|
|
||||||
|
assert status.status == 1
|
||||||
|
|
||||||
|
|
||||||
|
class TestMoonrakerExists:
|
||||||
|
def test_returns_instances(self, monkeypatch) -> None:
|
||||||
|
fake = object()
|
||||||
|
monkeypatch.setattr("utils.common.get_instances", lambda *_a, **_k: [fake])
|
||||||
|
assert moonraker_exists() == [fake]
|
||||||
|
|
||||||
|
def test_warns_when_none(self, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr("utils.common.get_instances", lambda *_a, **_k: [])
|
||||||
|
assert moonraker_exists("SomeInstaller") == []
|
||||||
|
|
||||||
|
|
||||||
|
class TestTruncString:
|
||||||
|
@pytest.mark.parametrize(
|
||||||
|
"value,length,expected",
|
||||||
|
[
|
||||||
|
("short", 10, "short"),
|
||||||
|
("exactly seven", 20, "exactly seven"),
|
||||||
|
("much longer string", 10, "much lo..."),
|
||||||
|
("abcdef", 5, "ab..."),
|
||||||
|
],
|
||||||
|
)
|
||||||
|
def test_truncates(self, value: str, length: int, expected: str) -> None:
|
||||||
|
assert trunc_string(value, length) == expected
|
||||||
@@ -0,0 +1,103 @@
|
|||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
from pathlib import Path
|
||||||
|
|
||||||
|
from utils.config_utils import (
|
||||||
|
add_config_section,
|
||||||
|
add_config_section_at_top,
|
||||||
|
remove_config_section,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class _FakeInstance:
|
||||||
|
def __init__(self, cfg_file: Path):
|
||||||
|
self.cfg_file = cfg_file
|
||||||
|
|
||||||
|
|
||||||
|
def _write_cfg(path: Path, content: str) -> None:
|
||||||
|
path.write_text(content, encoding="utf-8")
|
||||||
|
|
||||||
|
|
||||||
|
class TestAddConfigSection:
|
||||||
|
def test_creates_section_and_options(self, tmp_path: Path) -> None:
|
||||||
|
cfg = tmp_path / "printer.cfg"
|
||||||
|
_write_cfg(cfg, "[existing]\noption: value\n")
|
||||||
|
instance = _FakeInstance(cfg)
|
||||||
|
|
||||||
|
add_config_section(
|
||||||
|
"new_section",
|
||||||
|
[instance],
|
||||||
|
options=[("opt1", "val1"), ("opt2", ["line1", "line2"])],
|
||||||
|
)
|
||||||
|
|
||||||
|
text = cfg.read_text(encoding="utf-8")
|
||||||
|
assert "[new_section]" in text
|
||||||
|
assert "opt1: val1" in text
|
||||||
|
assert " line1" in text
|
||||||
|
assert " line2" in text
|
||||||
|
|
||||||
|
def test_skips_existing_section(self, tmp_path: Path) -> None:
|
||||||
|
cfg = tmp_path / "printer.cfg"
|
||||||
|
_write_cfg(cfg, "[section]\noption: value\n")
|
||||||
|
instance = _FakeInstance(cfg)
|
||||||
|
|
||||||
|
add_config_section("section", [instance])
|
||||||
|
|
||||||
|
text = cfg.read_text(encoding="utf-8")
|
||||||
|
assert text.count("[section]") == 1
|
||||||
|
|
||||||
|
def test_warns_when_file_missing(self, tmp_path: Path) -> None:
|
||||||
|
cfg = tmp_path / "missing.cfg"
|
||||||
|
instance = _FakeInstance(cfg)
|
||||||
|
|
||||||
|
add_config_section("section", [instance])
|
||||||
|
|
||||||
|
assert not cfg.exists()
|
||||||
|
|
||||||
|
|
||||||
|
class TestAddConfigSectionAtTop:
|
||||||
|
def test_prepends_section(self, tmp_path: Path) -> None:
|
||||||
|
cfg = tmp_path / "printer.cfg"
|
||||||
|
original = "[old]\noption: value\n"
|
||||||
|
_write_cfg(cfg, original)
|
||||||
|
instance = _FakeInstance(cfg)
|
||||||
|
|
||||||
|
add_config_section_at_top("top_section", [instance])
|
||||||
|
|
||||||
|
text = cfg.read_text(encoding="utf-8")
|
||||||
|
lines = text.splitlines()
|
||||||
|
assert lines[0] == "[top_section]"
|
||||||
|
assert "[old]" in text
|
||||||
|
assert text.endswith("\n")
|
||||||
|
|
||||||
|
|
||||||
|
class TestRemoveConfigSection:
|
||||||
|
def test_removes_existing(self, tmp_path: Path) -> None:
|
||||||
|
cfg = tmp_path / "printer.cfg"
|
||||||
|
_write_cfg(cfg, "[keep]\noption: 1\n[drop]\noption: 2\n")
|
||||||
|
instance = _FakeInstance(cfg)
|
||||||
|
|
||||||
|
removed = remove_config_section("drop", [instance])
|
||||||
|
|
||||||
|
assert removed == [instance]
|
||||||
|
text = cfg.read_text(encoding="utf-8")
|
||||||
|
assert "[drop]" not in text
|
||||||
|
assert "[keep]" in text
|
||||||
|
|
||||||
|
def test_skips_missing_section(self, tmp_path: Path) -> None:
|
||||||
|
cfg = tmp_path / "printer.cfg"
|
||||||
|
_write_cfg(cfg, "[keep]\noption: 1\n")
|
||||||
|
instance = _FakeInstance(cfg)
|
||||||
|
|
||||||
|
removed = remove_config_section("missing", [instance])
|
||||||
|
|
||||||
|
assert removed == []
|
||||||
|
assert cfg.read_text(encoding="utf-8") == "[keep]\noption: 1\n"
|
||||||
|
|
||||||
|
def test_warns_when_file_missing(self, tmp_path: Path) -> None:
|
||||||
|
cfg = tmp_path / "missing.cfg"
|
||||||
|
instance = _FakeInstance(cfg)
|
||||||
|
|
||||||
|
removed = remove_config_section("section", [instance])
|
||||||
|
|
||||||
|
assert removed == []
|
||||||
@@ -0,0 +1,245 @@
|
|||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
from pathlib import Path
|
||||||
|
from subprocess import CalledProcessError
|
||||||
|
from typing import Any, List
|
||||||
|
from zipfile import ZipFile
|
||||||
|
|
||||||
|
import pytest
|
||||||
|
from utils.fs_utils import (
|
||||||
|
check_file_exist,
|
||||||
|
create_folders,
|
||||||
|
create_symlink,
|
||||||
|
get_data_dir,
|
||||||
|
remove_file,
|
||||||
|
remove_with_sudo,
|
||||||
|
run_remove_routines,
|
||||||
|
unzip,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class TestCheckFileExist:
|
||||||
|
def test_returns_true_for_existing_file(self, tmp_path: Path) -> None:
|
||||||
|
file = tmp_path / "file.txt"
|
||||||
|
file.write_text("x")
|
||||||
|
assert check_file_exist(file) is True
|
||||||
|
|
||||||
|
def test_returns_false_for_missing_file(self, tmp_path: Path) -> None:
|
||||||
|
assert check_file_exist(tmp_path / "missing") is False
|
||||||
|
|
||||||
|
def test_returns_false_for_broken_symlink(self, tmp_path: Path) -> None:
|
||||||
|
link = tmp_path / "link"
|
||||||
|
link.symlink_to(tmp_path / "target")
|
||||||
|
assert check_file_exist(link) is False
|
||||||
|
|
||||||
|
def test_with_sudo_uses_subprocess(self, monkeypatch) -> None:
|
||||||
|
calls: List[List[str]] = []
|
||||||
|
|
||||||
|
def fake_check_output(cmd: List[str], **kwargs: Any) -> bytes:
|
||||||
|
calls.append(cmd)
|
||||||
|
return b""
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.fs_utils.check_output", fake_check_output)
|
||||||
|
path = Path("/some/path")
|
||||||
|
assert check_file_exist(path, sudo=True) is True
|
||||||
|
assert calls[0] == ["sudo", "find", "-L", "/some/path", "-maxdepth", "0"]
|
||||||
|
|
||||||
|
def test_with_sudo_returns_false_on_error(self, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.fs_utils.check_output",
|
||||||
|
lambda *a, **k: (_ for _ in ()).throw(CalledProcessError(1, "find")),
|
||||||
|
)
|
||||||
|
assert check_file_exist(Path("/some/path"), sudo=True) is False
|
||||||
|
|
||||||
|
|
||||||
|
class TestCreateSymlink:
|
||||||
|
def test_calls_ln_with_correct_args(self, monkeypatch) -> None:
|
||||||
|
runs: List[List[str]] = []
|
||||||
|
|
||||||
|
def fake_run(cmd: List[str], **kwargs: Any) -> Any:
|
||||||
|
runs.append(cmd)
|
||||||
|
return None
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.fs_utils.run", fake_run)
|
||||||
|
create_symlink(Path("/src"), Path("/dst"))
|
||||||
|
assert runs == [["ln", "-sf", "/src", "/dst"]]
|
||||||
|
|
||||||
|
def test_uses_sudo(self, monkeypatch) -> None:
|
||||||
|
runs: List[List[str]] = []
|
||||||
|
|
||||||
|
def fake_run(cmd: List[str], **kwargs: Any) -> Any:
|
||||||
|
runs.append(cmd)
|
||||||
|
return None
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.fs_utils.run", fake_run)
|
||||||
|
create_symlink(Path("/src"), Path("/dst"), sudo=True)
|
||||||
|
assert runs == [["sudo", "ln", "-sf", "/src", "/dst"]]
|
||||||
|
|
||||||
|
def test_raises_on_failure(self, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.fs_utils.run",
|
||||||
|
lambda *a, **k: (_ for _ in ()).throw(CalledProcessError(1, "ln")),
|
||||||
|
)
|
||||||
|
with pytest.raises(CalledProcessError):
|
||||||
|
create_symlink(Path("/src"), Path("/dst"))
|
||||||
|
|
||||||
|
|
||||||
|
class TestRemoveWithSudo:
|
||||||
|
def test_removes_existing_files(self, monkeypatch) -> None:
|
||||||
|
calls: List[tuple] = []
|
||||||
|
|
||||||
|
def fake_call(cmd: List[str], **kwargs: Any) -> int:
|
||||||
|
calls.append(("call", cmd))
|
||||||
|
return 0
|
||||||
|
|
||||||
|
def fake_run(cmd: List[str], **kwargs: Any) -> Any:
|
||||||
|
calls.append(("run", cmd))
|
||||||
|
return None
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.fs_utils.call", fake_call)
|
||||||
|
monkeypatch.setattr("utils.fs_utils.run", fake_run)
|
||||||
|
|
||||||
|
result = remove_with_sudo(Path("/some/file"))
|
||||||
|
|
||||||
|
assert result is True
|
||||||
|
assert ("call", ["sudo", "find", "/some/file"]) in calls
|
||||||
|
assert ("run", ["sudo", "rm", "-rf", "/some/file"]) in calls
|
||||||
|
|
||||||
|
def test_skips_missing_files(self, monkeypatch) -> None:
|
||||||
|
def fake_call(cmd: List[str], **kwargs: Any) -> int:
|
||||||
|
return 1
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.fs_utils.call", fake_call)
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.fs_utils.run",
|
||||||
|
lambda *a, **k: pytest.fail("should not run rm for missing file"),
|
||||||
|
)
|
||||||
|
|
||||||
|
assert remove_with_sudo(Path("/some/file")) is False
|
||||||
|
|
||||||
|
def test_accepts_list(self, monkeypatch) -> None:
|
||||||
|
runs: List[List[str]] = []
|
||||||
|
|
||||||
|
def fake_call(cmd: List[str], **kwargs: Any) -> int:
|
||||||
|
return 0
|
||||||
|
|
||||||
|
def fake_run(cmd: List[str], **kwargs: Any) -> Any:
|
||||||
|
runs.append(cmd)
|
||||||
|
return None
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.fs_utils.call", fake_call)
|
||||||
|
monkeypatch.setattr("utils.fs_utils.run", fake_run)
|
||||||
|
|
||||||
|
remove_with_sudo([Path("/a"), Path("/b")])
|
||||||
|
|
||||||
|
assert runs == [
|
||||||
|
["sudo", "rm", "-rf", "/a"],
|
||||||
|
["sudo", "rm", "-rf", "/b"],
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
|
class TestRemoveFile:
|
||||||
|
def test_calls_shell_rm(self, monkeypatch) -> None:
|
||||||
|
runs: List[Any] = []
|
||||||
|
|
||||||
|
def fake_run(cmd: str, **kwargs: Any) -> Any:
|
||||||
|
runs.append((cmd, kwargs.get("shell")))
|
||||||
|
return None
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.fs_utils.run", fake_run)
|
||||||
|
|
||||||
|
with pytest.warns(DeprecationWarning):
|
||||||
|
remove_file(Path("/some/file"), sudo=True)
|
||||||
|
|
||||||
|
assert runs == [("sudo rm -f /some/file", True)]
|
||||||
|
|
||||||
|
|
||||||
|
class TestRunRemoveRoutines:
|
||||||
|
def test_returns_false_for_missing(self, tmp_path: Path) -> None:
|
||||||
|
assert run_remove_routines(tmp_path / "missing") is False
|
||||||
|
|
||||||
|
def test_removes_file(self, tmp_path: Path) -> None:
|
||||||
|
file = tmp_path / "file.txt"
|
||||||
|
file.write_text("x")
|
||||||
|
assert run_remove_routines(file) is True
|
||||||
|
assert not file.exists()
|
||||||
|
|
||||||
|
def test_removes_directory(self, tmp_path: Path) -> None:
|
||||||
|
directory = tmp_path / "dir"
|
||||||
|
directory.mkdir()
|
||||||
|
(directory / "child").write_text("x")
|
||||||
|
assert run_remove_routines(directory) is True
|
||||||
|
assert not directory.exists()
|
||||||
|
|
||||||
|
def test_removes_symlink(self, tmp_path: Path) -> None:
|
||||||
|
target = tmp_path / "target"
|
||||||
|
target.write_text("x")
|
||||||
|
link = tmp_path / "link"
|
||||||
|
link.symlink_to(target)
|
||||||
|
assert run_remove_routines(link) is True
|
||||||
|
assert not link.exists()
|
||||||
|
assert target.exists()
|
||||||
|
|
||||||
|
|
||||||
|
class TestUnzip:
|
||||||
|
def test_extracts_contents(self, tmp_path: Path) -> None:
|
||||||
|
archive = tmp_path / "archive.zip"
|
||||||
|
target = tmp_path / "out"
|
||||||
|
target.mkdir()
|
||||||
|
|
||||||
|
with ZipFile(archive, "w") as zf:
|
||||||
|
zf.writestr("hello.txt", "world")
|
||||||
|
|
||||||
|
unzip(archive, target)
|
||||||
|
|
||||||
|
assert (target / "hello.txt").read_text() == "world"
|
||||||
|
|
||||||
|
|
||||||
|
class TestCreateFolders:
|
||||||
|
def test_creates_missing_directories(self, tmp_path: Path) -> None:
|
||||||
|
dirs = [tmp_path / "a", tmp_path / "b"]
|
||||||
|
create_folders(dirs)
|
||||||
|
assert all(d.exists() for d in dirs)
|
||||||
|
|
||||||
|
def test_skips_existing(self, tmp_path: Path) -> None:
|
||||||
|
existing = tmp_path / "exists"
|
||||||
|
existing.mkdir()
|
||||||
|
create_folders([existing])
|
||||||
|
assert existing.exists()
|
||||||
|
|
||||||
|
|
||||||
|
class TestGetDataDir:
|
||||||
|
def test_reads_from_service_file(self, tmp_path: Path, monkeypatch) -> None:
|
||||||
|
service = tmp_path / "klipper.service"
|
||||||
|
service.write_text(
|
||||||
|
"EnvironmentFile=/home/user/printer_data/systemd/klipper.env\n"
|
||||||
|
)
|
||||||
|
|
||||||
|
def fake_service_path(instance_type: type, suffix: str) -> Path:
|
||||||
|
return service
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.sys_utils.get_service_file_path", fake_service_path)
|
||||||
|
monkeypatch.setattr("utils.fs_utils.Path.home", lambda: tmp_path / "home")
|
||||||
|
|
||||||
|
result = get_data_dir(object, "")
|
||||||
|
assert result == Path("/home/user/printer_data")
|
||||||
|
|
||||||
|
def test_falls_back_to_suffixed_data_dir(self, tmp_path: Path, monkeypatch) -> None:
|
||||||
|
def fake_service_path(instance_type: type, suffix: str) -> Path:
|
||||||
|
return tmp_path / "no-such.service"
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.sys_utils.get_service_file_path", fake_service_path)
|
||||||
|
home = tmp_path / "home"
|
||||||
|
monkeypatch.setattr("utils.fs_utils.Path.home", lambda: home)
|
||||||
|
|
||||||
|
assert get_data_dir(object, "1") == home / "printer_1_data"
|
||||||
|
|
||||||
|
def test_falls_back_to_default_data_dir(self, tmp_path: Path, monkeypatch) -> None:
|
||||||
|
def fake_service_path(instance_type: type, suffix: str) -> Path:
|
||||||
|
return tmp_path / "no-such.service"
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.sys_utils.get_service_file_path", fake_service_path)
|
||||||
|
home = tmp_path / "home"
|
||||||
|
monkeypatch.setattr("utils.fs_utils.Path.home", lambda: home)
|
||||||
|
|
||||||
|
assert get_data_dir(object, "") == home / "printer_data"
|
||||||
@@ -0,0 +1,535 @@
|
|||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
from pathlib import Path
|
||||||
|
from subprocess import CalledProcessError
|
||||||
|
from typing import Any, List
|
||||||
|
|
||||||
|
import pytest
|
||||||
|
from utils.git_utils import (
|
||||||
|
GitException,
|
||||||
|
compare_semver_tags,
|
||||||
|
get_current_branch,
|
||||||
|
get_latest_remote_tag,
|
||||||
|
get_latest_unstable_tag,
|
||||||
|
get_local_commit,
|
||||||
|
get_local_tags,
|
||||||
|
get_remote_commit,
|
||||||
|
get_remote_tags,
|
||||||
|
get_repo_name,
|
||||||
|
get_repo_url,
|
||||||
|
git_clone_wrapper,
|
||||||
|
git_cmd_checkout,
|
||||||
|
git_cmd_clone,
|
||||||
|
git_cmd_pull,
|
||||||
|
git_pull_wrapper,
|
||||||
|
rollback_repository,
|
||||||
|
)
|
||||||
|
from utils.instance_type import InstanceType
|
||||||
|
|
||||||
|
|
||||||
|
class TestGitCmdPull:
|
||||||
|
def test_missing_dir_raises(self, tmp_path: Path) -> None:
|
||||||
|
missing = tmp_path / "does-not-exist"
|
||||||
|
with pytest.raises(GitException):
|
||||||
|
git_cmd_pull(missing)
|
||||||
|
|
||||||
|
def test_dir_without_git_raises(self, tmp_path: Path) -> None:
|
||||||
|
empty = tmp_path / "no-git"
|
||||||
|
empty.mkdir()
|
||||||
|
with pytest.raises(GitException):
|
||||||
|
git_cmd_pull(empty)
|
||||||
|
|
||||||
|
def test_success_runs_git_pull(self, monkeypatch) -> None:
|
||||||
|
repo = Path("/fake/repo")
|
||||||
|
runs: List[List[str]] = []
|
||||||
|
|
||||||
|
def fake_run(cmd: List[str], **kwargs: Any) -> Any:
|
||||||
|
runs.append(cmd)
|
||||||
|
return None
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.git_utils.Path.exists", lambda self: True)
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.git_utils.Path.joinpath", lambda self, name: repo / name
|
||||||
|
)
|
||||||
|
monkeypatch.setattr("utils.git_utils.run", fake_run)
|
||||||
|
|
||||||
|
git_cmd_pull(repo)
|
||||||
|
assert runs == [["git", "pull"]]
|
||||||
|
|
||||||
|
|
||||||
|
class TestGitCmdCheckout:
|
||||||
|
def test_missing_dir_raises(self, tmp_path: Path) -> None:
|
||||||
|
missing = tmp_path / "does-not-exist"
|
||||||
|
with pytest.raises(GitException):
|
||||||
|
git_cmd_checkout("main", missing)
|
||||||
|
|
||||||
|
def test_dir_without_git_raises(self, tmp_path: Path) -> None:
|
||||||
|
empty = tmp_path / "no-git"
|
||||||
|
empty.mkdir()
|
||||||
|
with pytest.raises(GitException):
|
||||||
|
git_cmd_checkout("main", empty)
|
||||||
|
|
||||||
|
def test_none_branch_returns(self, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.git_utils.run",
|
||||||
|
lambda *a, **k: pytest.fail("should not run checkout for None branch"),
|
||||||
|
)
|
||||||
|
git_cmd_checkout(None, Path("/repo"))
|
||||||
|
|
||||||
|
|
||||||
|
class TestGitPullWrapper:
|
||||||
|
def test_missing_dir_does_not_raise(self, tmp_path: Path) -> None:
|
||||||
|
missing = tmp_path / "does-not-exist"
|
||||||
|
git_pull_wrapper(missing)
|
||||||
|
|
||||||
|
def test_dir_without_git_does_not_raise(self, tmp_path: Path) -> None:
|
||||||
|
empty = tmp_path / "no-git"
|
||||||
|
empty.mkdir()
|
||||||
|
git_pull_wrapper(empty)
|
||||||
|
|
||||||
|
def test_success_calls_git_pull(self, monkeypatch) -> None:
|
||||||
|
repo = Path("/fake/repo")
|
||||||
|
called: List[Path] = []
|
||||||
|
|
||||||
|
def fake_git_cmd_pull(path: Path) -> None:
|
||||||
|
called.append(path)
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.git_utils.git_cmd_pull", fake_git_cmd_pull)
|
||||||
|
git_pull_wrapper(repo)
|
||||||
|
assert called == [repo]
|
||||||
|
|
||||||
|
|
||||||
|
class TestRollbackRepository:
|
||||||
|
def test_missing_dir_raises(
|
||||||
|
self, tmp_path: Path, monkeypatch: pytest.MonkeyPatch
|
||||||
|
) -> None:
|
||||||
|
missing = tmp_path / "does-not-exist"
|
||||||
|
called: list[bool] = []
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.git_utils.get_number_input",
|
||||||
|
lambda *_a, **_k: called.append(True) or 1,
|
||||||
|
)
|
||||||
|
with pytest.raises(GitException):
|
||||||
|
rollback_repository(missing, InstanceType)
|
||||||
|
assert not called
|
||||||
|
|
||||||
|
def test_dir_without_git_raises(
|
||||||
|
self, tmp_path: Path, monkeypatch: pytest.MonkeyPatch
|
||||||
|
) -> None:
|
||||||
|
empty = tmp_path / "no-git"
|
||||||
|
empty.mkdir()
|
||||||
|
called: list[bool] = []
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.git_utils.get_number_input",
|
||||||
|
lambda *_a, **_k: called.append(True) or 1,
|
||||||
|
)
|
||||||
|
with pytest.raises(GitException):
|
||||||
|
rollback_repository(empty, InstanceType)
|
||||||
|
assert not called
|
||||||
|
|
||||||
|
def test_aborts_when_not_confirmed(self, tmp_path: Path, monkeypatch) -> None:
|
||||||
|
repo = tmp_path / "repo"
|
||||||
|
repo.mkdir()
|
||||||
|
(repo / ".git").mkdir()
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.git_utils.get_number_input", lambda *a, **k: 2)
|
||||||
|
monkeypatch.setattr("utils.git_utils.get_confirm", lambda *a, **k: False)
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.git_utils.get_instances", lambda *a, **k: ["instance"]
|
||||||
|
)
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.git_utils.InstanceManager.stop_all",
|
||||||
|
lambda *a, **k: pytest.fail("should not stop when aborted"),
|
||||||
|
)
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.git_utils.run",
|
||||||
|
lambda *a, **k: pytest.fail("should not reset when aborted"),
|
||||||
|
)
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.git_utils.InstanceManager.start_all",
|
||||||
|
lambda *a, **k: pytest.fail("should not start when aborted"),
|
||||||
|
)
|
||||||
|
|
||||||
|
rollback_repository(repo, InstanceType)
|
||||||
|
|
||||||
|
def test_resets_and_restarts_when_confirmed(self, tmp_path: Path, monkeypatch) -> None:
|
||||||
|
repo = tmp_path / "repo"
|
||||||
|
repo.mkdir()
|
||||||
|
(repo / ".git").mkdir()
|
||||||
|
|
||||||
|
stops: List[List[Any]] = []
|
||||||
|
starts: List[List[Any]] = []
|
||||||
|
resets: List[List[str]] = []
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.git_utils.get_number_input", lambda *a, **k: 3)
|
||||||
|
monkeypatch.setattr("utils.git_utils.get_confirm", lambda *a, **k: True)
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.git_utils.get_instances", lambda *a, **k: ["instance"]
|
||||||
|
)
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.git_utils.InstanceManager.stop_all",
|
||||||
|
lambda instances: stops.append(instances),
|
||||||
|
)
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.git_utils.InstanceManager.start_all",
|
||||||
|
lambda instances: starts.append(instances),
|
||||||
|
)
|
||||||
|
|
||||||
|
def fake_run(cmd: List[str], **kwargs: Any) -> Any:
|
||||||
|
resets.append(cmd)
|
||||||
|
return None
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.git_utils.run", fake_run)
|
||||||
|
|
||||||
|
rollback_repository(repo, InstanceType)
|
||||||
|
|
||||||
|
assert stops == [["instance"]]
|
||||||
|
assert resets == [["git", "reset", "--hard", "HEAD~3"]]
|
||||||
|
assert starts == [["instance"]]
|
||||||
|
|
||||||
|
|
||||||
|
class TestGetRepoName:
|
||||||
|
def test_extracts_org_and_repo(self, monkeypatch, tmp_path: Path) -> None:
|
||||||
|
repo = tmp_path / "repo"
|
||||||
|
repo.mkdir()
|
||||||
|
(repo / ".git").mkdir()
|
||||||
|
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.git_utils.check_output",
|
||||||
|
lambda *a, **k: b"https://github.com/dw-0/kiauh.git\n",
|
||||||
|
)
|
||||||
|
assert get_repo_name(repo) == ("dw-0", "kiauh")
|
||||||
|
|
||||||
|
def test_returns_none_for_missing_repo(self, tmp_path: Path) -> None:
|
||||||
|
assert get_repo_name(tmp_path / "missing") == (None, None)
|
||||||
|
|
||||||
|
def test_returns_none_on_git_error(self, monkeypatch, tmp_path: Path) -> None:
|
||||||
|
repo = tmp_path / "repo"
|
||||||
|
repo.mkdir()
|
||||||
|
(repo / ".git").mkdir()
|
||||||
|
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.git_utils.check_output",
|
||||||
|
lambda *a, **k: (_ for _ in ()).throw(CalledProcessError(1, "git")),
|
||||||
|
)
|
||||||
|
assert get_repo_name(repo) == (None, None)
|
||||||
|
|
||||||
|
|
||||||
|
class TestGetCurrentBranch:
|
||||||
|
def test_returns_branch(self, monkeypatch, tmp_path: Path) -> None:
|
||||||
|
repo = tmp_path / "repo"
|
||||||
|
repo.mkdir()
|
||||||
|
(repo / ".git").mkdir()
|
||||||
|
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.git_utils.check_output", lambda *a, **k: b"feature-x\n"
|
||||||
|
)
|
||||||
|
assert get_current_branch(repo) == "feature-x"
|
||||||
|
|
||||||
|
def test_returns_none_for_missing_repo(self, tmp_path: Path) -> None:
|
||||||
|
assert get_current_branch(tmp_path / "missing") is None
|
||||||
|
|
||||||
|
|
||||||
|
class TestGetLocalTags:
|
||||||
|
def test_sorts_semver(self, monkeypatch, tmp_path: Path) -> None:
|
||||||
|
repo = tmp_path / "repo"
|
||||||
|
repo.mkdir()
|
||||||
|
(repo / ".git").mkdir()
|
||||||
|
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.git_utils.check_output",
|
||||||
|
lambda *a, **k: b"v1.0.0\nv1.0.1\nv1.0.10\nv1.0.2\nv2.0.0-beta.1\n",
|
||||||
|
)
|
||||||
|
assert get_local_tags(repo) == [
|
||||||
|
"v1.0.0",
|
||||||
|
"v1.0.1",
|
||||||
|
"v1.0.2",
|
||||||
|
"v1.0.10",
|
||||||
|
"v2.0.0-beta.1",
|
||||||
|
]
|
||||||
|
|
||||||
|
def test_returns_empty_for_missing_repo(self, tmp_path: Path) -> None:
|
||||||
|
assert get_local_tags(tmp_path / "missing") == []
|
||||||
|
|
||||||
|
|
||||||
|
class _FakeResponse:
|
||||||
|
def __init__(self, code: int, body: bytes = b""):
|
||||||
|
self._code = code
|
||||||
|
self._body = body
|
||||||
|
|
||||||
|
def __enter__(self):
|
||||||
|
return self
|
||||||
|
|
||||||
|
def __exit__(self, *args):
|
||||||
|
return None
|
||||||
|
|
||||||
|
def getcode(self) -> int:
|
||||||
|
return self._code
|
||||||
|
|
||||||
|
def read(self) -> bytes:
|
||||||
|
return self._body
|
||||||
|
|
||||||
|
|
||||||
|
class TestGetRemoteTags:
|
||||||
|
def test_parses_github_api(self, monkeypatch) -> None:
|
||||||
|
body = b'[{"name":"v1.0.0"},{"name":"v1.1.0"}]'
|
||||||
|
|
||||||
|
class FakeUrlLib:
|
||||||
|
@staticmethod
|
||||||
|
def urlopen(url: str):
|
||||||
|
return _FakeResponse(200, body)
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.git_utils.urllib.request", FakeUrlLib())
|
||||||
|
assert get_remote_tags("dw-0/kiauh") == ["v1.0.0", "v1.1.0"]
|
||||||
|
|
||||||
|
def test_returns_empty_on_http_error(self, monkeypatch) -> None:
|
||||||
|
class FakeUrlLib:
|
||||||
|
@staticmethod
|
||||||
|
def urlopen(url: str):
|
||||||
|
return _FakeResponse(404)
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.git_utils.urllib.request", FakeUrlLib())
|
||||||
|
assert get_remote_tags("dw-0/kiauh") == []
|
||||||
|
|
||||||
|
|
||||||
|
class TestGetLatestRemoteTag:
|
||||||
|
def test_returns_first_tag(self, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.git_utils.get_remote_tags", lambda *_a, **_k: ["v2.0.0", "v1.0.0"]
|
||||||
|
)
|
||||||
|
assert get_latest_remote_tag("dw-0/kiauh") == "v2.0.0"
|
||||||
|
|
||||||
|
def test_returns_empty_when_no_tags(self, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr("utils.git_utils.get_remote_tags", lambda *_a, **_k: [])
|
||||||
|
assert get_latest_remote_tag("dw-0/kiauh") == ""
|
||||||
|
|
||||||
|
|
||||||
|
class TestGetLatestUnstableTag:
|
||||||
|
def test_filters_prereleases(self, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.git_utils.get_remote_tags",
|
||||||
|
lambda *_a, **_k: ["v2.0.0", "v2.0.0-rc.1", "v1.0.0-beta.2"],
|
||||||
|
)
|
||||||
|
assert get_latest_unstable_tag("dw-0/kiauh") == "v2.0.0-rc.1"
|
||||||
|
|
||||||
|
def test_returns_empty_when_stable_only(self, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.git_utils.get_remote_tags", lambda *_a, **_k: ["v2.0.0", "v1.0.0"]
|
||||||
|
)
|
||||||
|
assert get_latest_unstable_tag("dw-0/kiauh") == ""
|
||||||
|
|
||||||
|
|
||||||
|
class TestCompareSemverTags:
|
||||||
|
@pytest.mark.parametrize(
|
||||||
|
"tag1,tag2,expected",
|
||||||
|
[
|
||||||
|
("v1.0.0", "v1.0.1", False),
|
||||||
|
("v1.1.0", "v1.0.1", True),
|
||||||
|
("v1.0.0", "v1.0.0", False),
|
||||||
|
("v2.0.0", "v1.9.9", True),
|
||||||
|
],
|
||||||
|
)
|
||||||
|
def test_comparison(self, tag1: str, tag2: str, expected: bool) -> None:
|
||||||
|
assert compare_semver_tags(tag1, tag2) is expected
|
||||||
|
|
||||||
|
|
||||||
|
class TestGetLocalCommit:
|
||||||
|
def test_describes_head(self, monkeypatch, tmp_path: Path) -> None:
|
||||||
|
repo = tmp_path / "repo"
|
||||||
|
repo.mkdir()
|
||||||
|
(repo / ".git").mkdir()
|
||||||
|
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.git_utils.check_output",
|
||||||
|
lambda *a, **k: "v1.0.0-0-gabc1234",
|
||||||
|
)
|
||||||
|
assert get_local_commit(repo) == "v1.0.0-0-gabc1234"
|
||||||
|
|
||||||
|
def test_returns_none_for_missing_repo(self, tmp_path: Path) -> None:
|
||||||
|
assert get_local_commit(tmp_path / "missing") is None
|
||||||
|
|
||||||
|
|
||||||
|
class TestGetRemoteCommit:
|
||||||
|
def test_describes_origin(self, monkeypatch, tmp_path: Path) -> None:
|
||||||
|
repo = tmp_path / "repo"
|
||||||
|
repo.mkdir()
|
||||||
|
(repo / ".git").mkdir()
|
||||||
|
|
||||||
|
def fake_check_output(cmd: str, **kwargs: Any) -> str:
|
||||||
|
if "HEAD" in cmd:
|
||||||
|
return "v1.0.0"
|
||||||
|
return "origin/main"
|
||||||
|
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.git_utils.get_current_branch", lambda *_a, **_k: "main"
|
||||||
|
)
|
||||||
|
monkeypatch.setattr("utils.git_utils.check_output", fake_check_output)
|
||||||
|
assert get_remote_commit(repo) == "origin/main"
|
||||||
|
|
||||||
|
def test_returns_none_for_missing_repo(self, tmp_path: Path) -> None:
|
||||||
|
assert get_remote_commit(tmp_path / "missing") is None
|
||||||
|
|
||||||
|
|
||||||
|
class TestGitCmdClone:
|
||||||
|
def test_without_blobless(self, monkeypatch) -> None:
|
||||||
|
runs: List[List[str]] = []
|
||||||
|
|
||||||
|
def fake_run(cmd: List[str], **kwargs: Any) -> Any:
|
||||||
|
runs.append(cmd)
|
||||||
|
return None
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.git_utils.run", fake_run)
|
||||||
|
git_cmd_clone("https://github.com/dw-0/kiauh", Path("/target"))
|
||||||
|
assert runs == [["git", "clone", "https://github.com/dw-0/kiauh", "/target"]]
|
||||||
|
|
||||||
|
def test_with_blobless(self, monkeypatch) -> None:
|
||||||
|
runs: List[List[str]] = []
|
||||||
|
|
||||||
|
def fake_run(cmd: List[str], **kwargs: Any) -> Any:
|
||||||
|
runs.append(cmd)
|
||||||
|
return None
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.git_utils.run", fake_run)
|
||||||
|
git_cmd_clone(
|
||||||
|
"https://github.com/dw-0/kiauh", Path("/target"), blobless=True
|
||||||
|
)
|
||||||
|
assert runs == [
|
||||||
|
[
|
||||||
|
"git",
|
||||||
|
"clone",
|
||||||
|
"--filter=blob:none",
|
||||||
|
"https://github.com/dw-0/kiauh",
|
||||||
|
"/target",
|
||||||
|
]
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
|
class TestGitCmdCheckoutSingle:
|
||||||
|
def test_runs_git_checkout(self, monkeypatch, tmp_path: Path) -> None:
|
||||||
|
repo = tmp_path / "repo"
|
||||||
|
repo.mkdir()
|
||||||
|
(repo / ".git").mkdir()
|
||||||
|
runs: List[List[str]] = []
|
||||||
|
|
||||||
|
def fake_run(cmd: List[str], **kwargs: Any) -> Any:
|
||||||
|
runs.append(cmd)
|
||||||
|
return None
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.git_utils.run", fake_run)
|
||||||
|
git_cmd_checkout("dev", repo)
|
||||||
|
assert runs == [["git", "checkout", "dev"]]
|
||||||
|
|
||||||
|
|
||||||
|
class TestGetRepoUrl:
|
||||||
|
def test_extracts_remote_url(self, monkeypatch, tmp_path: Path) -> None:
|
||||||
|
repo = tmp_path / "repo"
|
||||||
|
repo.mkdir()
|
||||||
|
(repo / ".git").mkdir()
|
||||||
|
|
||||||
|
class FakeResult:
|
||||||
|
stdout = "https://github.com/dw-0/kiauh.git\n"
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.git_utils.run", lambda *a, **k: FakeResult())
|
||||||
|
assert get_repo_url(repo) == "https://github.com/dw-0/kiauh.git"
|
||||||
|
|
||||||
|
def test_returns_none_for_missing_repo(self, tmp_path: Path) -> None:
|
||||||
|
assert get_repo_url(tmp_path / "missing") is None
|
||||||
|
|
||||||
|
def test_returns_none_on_git_error(self, monkeypatch, tmp_path: Path) -> None:
|
||||||
|
repo = tmp_path / "repo"
|
||||||
|
repo.mkdir()
|
||||||
|
(repo / ".git").mkdir()
|
||||||
|
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.git_utils.run",
|
||||||
|
lambda *a, **k: (_ for _ in ()).throw(CalledProcessError(1, "git")),
|
||||||
|
)
|
||||||
|
assert get_repo_url(repo) is None
|
||||||
|
|
||||||
|
|
||||||
|
class _CloneRecorder:
|
||||||
|
def __init__(self):
|
||||||
|
self.calls: List[tuple] = []
|
||||||
|
self.checkouts: List[tuple] = []
|
||||||
|
self.removed: List[Path] = []
|
||||||
|
|
||||||
|
def fake_clone(self, repo: str, target: Path, blobless: bool = False) -> None:
|
||||||
|
self.calls.append((repo, target, blobless))
|
||||||
|
|
||||||
|
def fake_checkout(self, branch: str | None, target: Path) -> None:
|
||||||
|
self.checkouts.append((branch, target))
|
||||||
|
|
||||||
|
|
||||||
|
class TestGitCloneWrapper:
|
||||||
|
def test_clones_when_target_missing(self, monkeypatch, tmp_path: Path) -> None:
|
||||||
|
target = tmp_path / "kiauh"
|
||||||
|
recorder = _CloneRecorder()
|
||||||
|
monkeypatch.setattr("utils.git_utils.git_cmd_clone", recorder.fake_clone)
|
||||||
|
monkeypatch.setattr("utils.git_utils.git_cmd_checkout", recorder.fake_checkout)
|
||||||
|
|
||||||
|
git_clone_wrapper("https://github.com/dw-0/kiauh", target, branch="dev")
|
||||||
|
|
||||||
|
assert recorder.calls == [("https://github.com/dw-0/kiauh", target, True)]
|
||||||
|
assert recorder.checkouts == [("dev", target)]
|
||||||
|
|
||||||
|
def test_skips_checkout_for_main(self, monkeypatch, tmp_path: Path) -> None:
|
||||||
|
target = tmp_path / "kiauh"
|
||||||
|
recorder = _CloneRecorder()
|
||||||
|
monkeypatch.setattr("utils.git_utils.git_cmd_clone", recorder.fake_clone)
|
||||||
|
monkeypatch.setattr("utils.git_utils.git_cmd_checkout", recorder.fake_checkout)
|
||||||
|
|
||||||
|
git_clone_wrapper("https://github.com/dw-0/kiauh", target, branch="main")
|
||||||
|
|
||||||
|
assert recorder.checkouts == []
|
||||||
|
|
||||||
|
def test_prompts_before_overwrite(self, monkeypatch, tmp_path: Path) -> None:
|
||||||
|
target = tmp_path / "kiauh"
|
||||||
|
target.mkdir()
|
||||||
|
recorder = _CloneRecorder()
|
||||||
|
removed: List[Path] = []
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.git_utils.git_cmd_clone", recorder.fake_clone)
|
||||||
|
monkeypatch.setattr("utils.git_utils.git_cmd_checkout", recorder.fake_checkout)
|
||||||
|
monkeypatch.setattr("utils.git_utils.shutil.rmtree", lambda p: removed.append(p))
|
||||||
|
monkeypatch.setattr("utils.git_utils.get_confirm", lambda *a, **k: True)
|
||||||
|
|
||||||
|
git_clone_wrapper("https://github.com/dw-0/kiauh", target, branch="dev")
|
||||||
|
|
||||||
|
assert removed == [target]
|
||||||
|
assert recorder.calls == [("https://github.com/dw-0/kiauh", target, True)]
|
||||||
|
|
||||||
|
def test_respects_decline_to_overwrite(self, monkeypatch, tmp_path: Path) -> None:
|
||||||
|
target = tmp_path / "kiauh"
|
||||||
|
target.mkdir()
|
||||||
|
recorder = _CloneRecorder()
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.git_utils.git_cmd_clone", recorder.fake_clone)
|
||||||
|
monkeypatch.setattr("utils.git_utils.git_cmd_checkout", recorder.fake_checkout)
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.git_utils.shutil.rmtree",
|
||||||
|
lambda *a, **k: pytest.fail("should not remove"),
|
||||||
|
)
|
||||||
|
monkeypatch.setattr("utils.git_utils.get_confirm", lambda *a, **k: False)
|
||||||
|
|
||||||
|
git_clone_wrapper("https://github.com/dw-0/kiauh", target)
|
||||||
|
|
||||||
|
assert recorder.calls == []
|
||||||
|
|
||||||
|
def test_force_overwrites_without_prompt(self, monkeypatch, tmp_path: Path) -> None:
|
||||||
|
target = tmp_path / "kiauh"
|
||||||
|
target.mkdir()
|
||||||
|
recorder = _CloneRecorder()
|
||||||
|
removed: List[Path] = []
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.git_utils.git_cmd_clone", recorder.fake_clone)
|
||||||
|
monkeypatch.setattr("utils.git_utils.git_cmd_checkout", recorder.fake_checkout)
|
||||||
|
monkeypatch.setattr("utils.git_utils.shutil.rmtree", lambda p: removed.append(p))
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.git_utils.get_confirm",
|
||||||
|
lambda *a, **k: pytest.fail("should not prompt when forced"),
|
||||||
|
)
|
||||||
|
|
||||||
|
git_clone_wrapper("https://github.com/dw-0/kiauh", target, force=True)
|
||||||
|
|
||||||
|
assert removed == [target]
|
||||||
|
assert recorder.calls == [("https://github.com/dw-0/kiauh", target, True)]
|
||||||
@@ -0,0 +1,158 @@
|
|||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
from typing import Any, List
|
||||||
|
|
||||||
|
import pytest
|
||||||
|
from utils.input_utils import (
|
||||||
|
format_question,
|
||||||
|
get_confirm,
|
||||||
|
get_number_input,
|
||||||
|
get_selection_input,
|
||||||
|
get_string_input,
|
||||||
|
validate_number_input,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def _input_sequence(answers: List[str]):
|
||||||
|
it = iter(answers)
|
||||||
|
|
||||||
|
def _input(_prompt: str = "") -> str:
|
||||||
|
return next(it)
|
||||||
|
|
||||||
|
return _input
|
||||||
|
|
||||||
|
|
||||||
|
class TestGetConfirm:
|
||||||
|
def test_accepts_yes(self, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr("builtins.input", _input_sequence(["y"]))
|
||||||
|
assert get_confirm("go?") is True
|
||||||
|
|
||||||
|
def test_accepts_no(self, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr("builtins.input", _input_sequence(["n"]))
|
||||||
|
assert get_confirm("go?") is False
|
||||||
|
|
||||||
|
def test_default_yes_on_empty(self, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr("builtins.input", _input_sequence([""]))
|
||||||
|
assert get_confirm("go?", default_choice=True) is True
|
||||||
|
|
||||||
|
def test_default_no_on_empty(self, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr("builtins.input", _input_sequence([""]))
|
||||||
|
assert get_confirm("go?", default_choice=False) is False
|
||||||
|
|
||||||
|
def test_handles_invalid_then_valid(self, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr("builtins.input", _input_sequence(["maybe", "yes"]))
|
||||||
|
assert get_confirm("go?") is True
|
||||||
|
|
||||||
|
def test_go_back_returns_none(self, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr("builtins.input", _input_sequence(["b"]))
|
||||||
|
assert get_confirm("go?", allow_go_back=True) is None
|
||||||
|
|
||||||
|
|
||||||
|
class TestGetNumberInput:
|
||||||
|
def test_returns_valid(self, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr("builtins.input", _input_sequence(["5"]))
|
||||||
|
assert get_number_input("count?", 1, 10) == 5
|
||||||
|
|
||||||
|
def test_uses_default(self, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr("builtins.input", _input_sequence([""]))
|
||||||
|
assert get_number_input("count?", 1, default=3) == 3
|
||||||
|
|
||||||
|
def test_enforces_minimum(self, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr("builtins.input", _input_sequence(["0", "2"]))
|
||||||
|
assert get_number_input("count?", 1) == 2
|
||||||
|
|
||||||
|
def test_enforces_maximum(self, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr("builtins.input", _input_sequence(["11", "9"]))
|
||||||
|
assert get_number_input("count?", 1, 10) == 9
|
||||||
|
|
||||||
|
def test_go_back_returns_none(self, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr("builtins.input", _input_sequence(["b"]))
|
||||||
|
assert get_number_input("count?", 1, allow_go_back=True) is None
|
||||||
|
|
||||||
|
|
||||||
|
class TestGetStringInput:
|
||||||
|
def test_accepts_alphanumeric(self, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr("builtins.input", _input_sequence(["abc123"]))
|
||||||
|
assert get_string_input("name?") == "abc123"
|
||||||
|
|
||||||
|
def test_rejects_empty(self, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr("builtins.input", _input_sequence(["", "value"]))
|
||||||
|
assert get_string_input("name?") == "value"
|
||||||
|
|
||||||
|
def test_uses_default(self, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr("builtins.input", _input_sequence([""]))
|
||||||
|
assert get_string_input("name?", default="fallback") == "fallback"
|
||||||
|
|
||||||
|
def test_validates_regex(self, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr("builtins.input", _input_sequence(["@", "#"]))
|
||||||
|
assert get_string_input("name?", regex=r"^#+$") == "#"
|
||||||
|
|
||||||
|
def test_rejects_excluded(self, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr("builtins.input", _input_sequence(["taken", "free"]))
|
||||||
|
assert get_string_input("name?", exclude=["taken"]) == "free"
|
||||||
|
|
||||||
|
def test_allows_special_chars(self, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr("builtins.input", _input_sequence(["a-b_c"]))
|
||||||
|
assert get_string_input("name?", allow_special_chars=True) == "a-b_c"
|
||||||
|
|
||||||
|
def test_allows_empty_with_special_chars(self, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr("builtins.input", _input_sequence([""]))
|
||||||
|
assert (
|
||||||
|
get_string_input("name?", allow_empty=True, allow_special_chars=True) == ""
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class TestGetSelectionInput:
|
||||||
|
def test_from_list(self, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr("builtins.input", _input_sequence(["b"]))
|
||||||
|
assert get_selection_input("pick?", ["a", "b", "c"]) == "b"
|
||||||
|
|
||||||
|
def test_from_dict(self, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr("builtins.input", _input_sequence(["two"]))
|
||||||
|
assert get_selection_input("pick?", {"one": 1, "two": 2}) == "two"
|
||||||
|
|
||||||
|
def test_invalid_then_valid(self, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr("builtins.input", _input_sequence(["z", "a"]))
|
||||||
|
assert get_selection_input("pick?", ["a", "b"]) == "a"
|
||||||
|
|
||||||
|
def test_invalid_type_raises(self, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr("builtins.input", _input_sequence(["x"]))
|
||||||
|
with pytest.raises(ValueError):
|
||||||
|
get_selection_input("pick?", 123) # type: ignore[arg-type]
|
||||||
|
|
||||||
|
|
||||||
|
class TestFormatQuestion:
|
||||||
|
def test_includes_default(self) -> None:
|
||||||
|
assert "default=5" in format_question("count", 5)
|
||||||
|
|
||||||
|
def test_no_default(self) -> None:
|
||||||
|
assert "count" in format_question("count")
|
||||||
|
assert "default" not in format_question("count")
|
||||||
|
|
||||||
|
|
||||||
|
class TestValidateNumberInput:
|
||||||
|
@pytest.mark.parametrize(
|
||||||
|
"value,min_count,max_count,expected",
|
||||||
|
[
|
||||||
|
("5", 1, 10, 5),
|
||||||
|
("1", 1, 10, 1),
|
||||||
|
("10", 1, 10, 10),
|
||||||
|
("3", 1, None, 3),
|
||||||
|
],
|
||||||
|
)
|
||||||
|
def test_valid(
|
||||||
|
self, value: str, min_count: int, max_count: Any, expected: int
|
||||||
|
) -> None:
|
||||||
|
assert validate_number_input(value, min_count, max_count) == expected
|
||||||
|
|
||||||
|
@pytest.mark.parametrize(
|
||||||
|
"value,min_count,max_count",
|
||||||
|
[
|
||||||
|
("0", 1, 10),
|
||||||
|
("11", 1, 10),
|
||||||
|
("-1", 0, None),
|
||||||
|
],
|
||||||
|
)
|
||||||
|
def test_raises(self, value: str, min_count: int, max_count: Any) -> None:
|
||||||
|
with pytest.raises(ValueError):
|
||||||
|
validate_number_input(value, min_count, max_count)
|
||||||
@@ -0,0 +1,17 @@
|
|||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
from typing import TypeVar
|
||||||
|
|
||||||
|
from components.klipper.klipper import Klipper
|
||||||
|
from components.moonraker.moonraker import Moonraker
|
||||||
|
from utils.instance_type import InstanceType
|
||||||
|
|
||||||
|
|
||||||
|
class TestInstanceType:
|
||||||
|
def test_is_typevar(self) -> None:
|
||||||
|
assert isinstance(InstanceType, TypeVar)
|
||||||
|
|
||||||
|
def test_bound_classes_include_components(self) -> None:
|
||||||
|
bound = InstanceType.__constraints__
|
||||||
|
assert Klipper in bound
|
||||||
|
assert Moonraker in bound
|
||||||
@@ -0,0 +1,98 @@
|
|||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
from pathlib import Path
|
||||||
|
from typing import List
|
||||||
|
|
||||||
|
import pytest
|
||||||
|
from utils.instance_utils import (
|
||||||
|
get_instance_suffix,
|
||||||
|
get_instances,
|
||||||
|
stop_klipper_instances_interactively,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class Klipper:
|
||||||
|
def __init__(self, suffix: str):
|
||||||
|
self.suffix = suffix
|
||||||
|
|
||||||
|
def __eq__(self, other):
|
||||||
|
return isinstance(other, Klipper) and self.suffix == other.suffix
|
||||||
|
|
||||||
|
def __repr__(self):
|
||||||
|
return f"Klipper({self.suffix!r})"
|
||||||
|
|
||||||
|
|
||||||
|
class TestGetInstances:
|
||||||
|
def test_returns_empty_when_no_services(self, tmp_path: Path, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr("utils.instance_utils.SYSTEMD", tmp_path)
|
||||||
|
assert get_instances(Klipper) == []
|
||||||
|
|
||||||
|
def test_raises_when_not_a_class(self, tmp_path: Path, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr("utils.instance_utils.SYSTEMD", tmp_path)
|
||||||
|
with pytest.raises(ValueError):
|
||||||
|
get_instances("not-a-class") # type: ignore[arg-type]
|
||||||
|
|
||||||
|
def test_finds_and_sorts_instances(self, tmp_path: Path, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr("utils.instance_utils.SYSTEMD", tmp_path)
|
||||||
|
|
||||||
|
(tmp_path / "klipper.service").write_text("")
|
||||||
|
(tmp_path / "klipper-1.service").write_text("")
|
||||||
|
(tmp_path / "klipper-10.service").write_text("")
|
||||||
|
(tmp_path / "klipper-a.service").write_text("")
|
||||||
|
|
||||||
|
instances = get_instances(Klipper)
|
||||||
|
assert [i.suffix for i in instances] == ["", "1", "10", "a"]
|
||||||
|
|
||||||
|
def test_excludes_blacklisted_suffixes(self, tmp_path: Path, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr("utils.instance_utils.SYSTEMD", tmp_path)
|
||||||
|
|
||||||
|
(tmp_path / "klipper.service").write_text("")
|
||||||
|
(tmp_path / "klipper-mcu.service").write_text("")
|
||||||
|
|
||||||
|
instances = get_instances(Klipper)
|
||||||
|
assert [i.suffix for i in instances] == [""]
|
||||||
|
|
||||||
|
|
||||||
|
class TestGetInstanceSuffix:
|
||||||
|
@pytest.mark.parametrize(
|
||||||
|
"name,service,expected",
|
||||||
|
[
|
||||||
|
("klipper", "klipper.service", ""),
|
||||||
|
("klipper", "klipper-1.service", "1"),
|
||||||
|
("klipper", "klipper-10.service", "10"),
|
||||||
|
("moonraker", "moonraker-foo.service", "foo"),
|
||||||
|
],
|
||||||
|
)
|
||||||
|
def test_suffix(self, name: str, service: str, expected: str) -> None:
|
||||||
|
assert get_instance_suffix(name, Path(service)) == expected
|
||||||
|
|
||||||
|
|
||||||
|
class TestStopKlipperInstancesInteractively:
|
||||||
|
def test_empty_returns_true(self, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.instance_utils.get_confirm",
|
||||||
|
lambda *a, **k: pytest.fail("no prompt when no instances"),
|
||||||
|
)
|
||||||
|
assert stop_klipper_instances_interactively([]) is True
|
||||||
|
|
||||||
|
def test_stops_on_confirm(self, monkeypatch) -> None:
|
||||||
|
stopped: List[Klipper] = []
|
||||||
|
instance = Klipper("")
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.instance_utils.get_confirm", lambda *a, **k: True)
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.instance_utils.InstanceManager.stop_all",
|
||||||
|
lambda instances: stopped.extend(instances),
|
||||||
|
)
|
||||||
|
|
||||||
|
assert stop_klipper_instances_interactively([instance], "update") is True
|
||||||
|
assert [i.suffix for i in stopped] == [""]
|
||||||
|
|
||||||
|
def test_aborts_on_decline(self, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr("utils.instance_utils.get_confirm", lambda *a, **k: False)
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.instance_utils.InstanceManager.stop_all",
|
||||||
|
lambda *a, **k: pytest.fail("should not stop when declined"),
|
||||||
|
)
|
||||||
|
|
||||||
|
assert stop_klipper_instances_interactively([Klipper("")]) is False
|
||||||
@@ -0,0 +1,686 @@
|
|||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import builtins
|
||||||
|
from io import StringIO
|
||||||
|
from pathlib import Path
|
||||||
|
from subprocess import CalledProcessError
|
||||||
|
from typing import Any, List
|
||||||
|
|
||||||
|
import pytest
|
||||||
|
from utils.sys_utils import (
|
||||||
|
VenvCreationFailedException,
|
||||||
|
check_package_install,
|
||||||
|
check_python_version,
|
||||||
|
cmd_sysctl_manage,
|
||||||
|
cmd_sysctl_service,
|
||||||
|
create_env_file,
|
||||||
|
create_python_venv,
|
||||||
|
create_service_file,
|
||||||
|
download_file,
|
||||||
|
download_progress,
|
||||||
|
get_distro_info,
|
||||||
|
get_ipv4_addr,
|
||||||
|
get_service_file_path,
|
||||||
|
get_system_timezone,
|
||||||
|
get_upgradable_packages,
|
||||||
|
install_python_packages,
|
||||||
|
install_python_requirements,
|
||||||
|
install_system_packages,
|
||||||
|
kill,
|
||||||
|
log_process,
|
||||||
|
parse_packages_from_file,
|
||||||
|
remove_system_service,
|
||||||
|
set_nginx_permissions,
|
||||||
|
unit_file_exists,
|
||||||
|
update_python_pip,
|
||||||
|
update_system_package_lists,
|
||||||
|
upgrade_system_packages,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class TestKill:
|
||||||
|
def test_exits_with_error(self, monkeypatch) -> None:
|
||||||
|
exited: List[int] = []
|
||||||
|
|
||||||
|
def fake_exit(code: int) -> None:
|
||||||
|
exited.append(code)
|
||||||
|
raise SystemExit(code)
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.sys_utils.sys.exit", fake_exit)
|
||||||
|
with pytest.raises(SystemExit):
|
||||||
|
kill("boom")
|
||||||
|
assert exited == [1]
|
||||||
|
|
||||||
|
|
||||||
|
class TestCheckPythonVersion:
|
||||||
|
def test_old(self, monkeypatch) -> None:
|
||||||
|
info = type("VI", (), {"major": 3, "minor": 7})()
|
||||||
|
monkeypatch.setattr("utils.sys_utils.sys.version_info", info)
|
||||||
|
assert check_python_version(3, 8) is False
|
||||||
|
|
||||||
|
def test_current(self, monkeypatch) -> None:
|
||||||
|
info = type("VI", (), {"major": 3, "minor": 9})()
|
||||||
|
monkeypatch.setattr("utils.sys_utils.sys.version_info", info)
|
||||||
|
assert check_python_version(3, 8) is True
|
||||||
|
|
||||||
|
|
||||||
|
class TestParsePackagesFromFile:
|
||||||
|
def test_reads_pkglist(self, tmp_path: Path) -> None:
|
||||||
|
script = tmp_path / "install.sh"
|
||||||
|
script.write_text('PKGLIST="git curl wget"\nOTHER="x"\n')
|
||||||
|
assert parse_packages_from_file(script) == ["git", "curl", "wget"]
|
||||||
|
|
||||||
|
|
||||||
|
class TestCreatePythonVenv:
|
||||||
|
def test_creates_when_missing(self, monkeypatch) -> None:
|
||||||
|
runs: List[List[str]] = []
|
||||||
|
|
||||||
|
def fake_run(cmd: List[str], **kwargs: Any) -> Any:
|
||||||
|
runs.append(cmd)
|
||||||
|
return None
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.sys_utils.run", fake_run)
|
||||||
|
target = Path("/tmp/venv")
|
||||||
|
|
||||||
|
assert create_python_venv(target) is True
|
||||||
|
assert runs == [
|
||||||
|
["virtualenv", "-p", "/usr/bin/python3", "/tmp/venv"],
|
||||||
|
]
|
||||||
|
|
||||||
|
def test_declines_recreate(self, monkeypatch) -> None:
|
||||||
|
target = Path("/tmp/venv")
|
||||||
|
monkeypatch.setattr("utils.sys_utils.Path.exists", lambda self: self == target)
|
||||||
|
monkeypatch.setattr("utils.sys_utils.get_confirm", lambda *a, **k: False)
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.sys_utils.run",
|
||||||
|
lambda *a, **k: pytest.fail("should not recreate when declined"),
|
||||||
|
)
|
||||||
|
|
||||||
|
assert create_python_venv(target) is False
|
||||||
|
|
||||||
|
def test_confirms_recreate(self, monkeypatch) -> None:
|
||||||
|
target = Path("/tmp/venv")
|
||||||
|
state = {"exists": True}
|
||||||
|
removed: List[Path] = []
|
||||||
|
runs: List[List[str]] = []
|
||||||
|
|
||||||
|
def fake_exists(self: Path) -> bool:
|
||||||
|
return state["exists"] and str(self) == str(target)
|
||||||
|
|
||||||
|
def fake_run(cmd: List[str], **kwargs: Any) -> Any:
|
||||||
|
runs.append(cmd)
|
||||||
|
return None
|
||||||
|
|
||||||
|
def fake_rmtree(p: Path) -> None:
|
||||||
|
removed.append(p)
|
||||||
|
state["exists"] = False
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.sys_utils.Path.exists", fake_exists)
|
||||||
|
monkeypatch.setattr("utils.sys_utils.get_confirm", lambda *a, **k: True)
|
||||||
|
monkeypatch.setattr("utils.sys_utils.shutil.rmtree", fake_rmtree)
|
||||||
|
monkeypatch.setattr("utils.sys_utils.run", fake_run)
|
||||||
|
|
||||||
|
assert (
|
||||||
|
create_python_venv(target, allow_access_to_system_site_packages=True)
|
||||||
|
is True
|
||||||
|
)
|
||||||
|
assert removed == [target]
|
||||||
|
assert runs == [
|
||||||
|
[
|
||||||
|
"virtualenv",
|
||||||
|
"-p",
|
||||||
|
"/usr/bin/python3",
|
||||||
|
"/tmp/venv",
|
||||||
|
"--system-site-packages",
|
||||||
|
],
|
||||||
|
]
|
||||||
|
|
||||||
|
def test_force_recreate(self, monkeypatch) -> None:
|
||||||
|
target = Path("/tmp/venv")
|
||||||
|
state = {"exists": True}
|
||||||
|
removed: List[Path] = []
|
||||||
|
|
||||||
|
def fake_exists(self: Path) -> bool:
|
||||||
|
return state["exists"] and str(self) == str(target)
|
||||||
|
|
||||||
|
def fake_rmtree(p: Path) -> None:
|
||||||
|
removed.append(p)
|
||||||
|
state["exists"] = False
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.sys_utils.Path.exists", fake_exists)
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.sys_utils.get_confirm",
|
||||||
|
lambda *a, **k: pytest.fail("should not prompt when forced"),
|
||||||
|
)
|
||||||
|
monkeypatch.setattr("utils.sys_utils.shutil.rmtree", fake_rmtree)
|
||||||
|
monkeypatch.setattr("utils.sys_utils.run", lambda *a, **k: None)
|
||||||
|
|
||||||
|
assert create_python_venv(target, force=True) is True
|
||||||
|
assert removed == [target]
|
||||||
|
|
||||||
|
def test_creation_failure(self, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.sys_utils.run",
|
||||||
|
lambda *a, **k: (_ for _ in ()).throw(CalledProcessError(1, "virtualenv")),
|
||||||
|
)
|
||||||
|
assert create_python_venv(Path("/tmp/venv")) is False
|
||||||
|
|
||||||
|
def test_remove_failure(self, monkeypatch) -> None:
|
||||||
|
target = Path("/tmp/venv")
|
||||||
|
|
||||||
|
def fake_exists(self: Path) -> bool:
|
||||||
|
return str(self) == str(target)
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.sys_utils.Path.exists", fake_exists)
|
||||||
|
monkeypatch.setattr("utils.sys_utils.get_confirm", lambda *a, **k: True)
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.sys_utils.shutil.rmtree",
|
||||||
|
lambda *a, **k: (_ for _ in ()).throw(OSError("locked")),
|
||||||
|
)
|
||||||
|
|
||||||
|
assert create_python_venv(target) is False
|
||||||
|
|
||||||
|
|
||||||
|
class TestUpdatePythonPip:
|
||||||
|
def test_raises_when_pip_missing(self, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr("utils.sys_utils.check_file_exist", lambda *a, **k: False)
|
||||||
|
with pytest.raises(FileNotFoundError):
|
||||||
|
update_python_pip(Path("/tmp/venv"))
|
||||||
|
|
||||||
|
def test_runs_upgrade(self, monkeypatch) -> None:
|
||||||
|
runs: List[List[str]] = []
|
||||||
|
|
||||||
|
def fake_run(cmd: List[str], **kwargs: Any) -> Any:
|
||||||
|
runs.append(cmd)
|
||||||
|
return type("R", (), {"returncode": 0, "stderr": ""})()
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.sys_utils.check_file_exist", lambda *a, **k: True)
|
||||||
|
monkeypatch.setattr("utils.sys_utils.run", fake_run)
|
||||||
|
|
||||||
|
update_python_pip(Path("/tmp/venv"))
|
||||||
|
assert runs == [["/tmp/venv/bin/pip", "install", "-U", "pip"]]
|
||||||
|
|
||||||
|
def test_logs_stderr(self, monkeypatch, capsys) -> None:
|
||||||
|
def fake_run(cmd: List[str], **kwargs: Any) -> Any:
|
||||||
|
return type("R", (), {"returncode": 0, "stderr": "some warning"})()
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.sys_utils.check_file_exist", lambda *a, **k: True)
|
||||||
|
monkeypatch.setattr("utils.sys_utils.run", fake_run)
|
||||||
|
|
||||||
|
update_python_pip(Path("/tmp/venv"))
|
||||||
|
|
||||||
|
|
||||||
|
class TestInstallPythonRequirements:
|
||||||
|
def test_success(self, monkeypatch) -> None:
|
||||||
|
def fake_run(cmd: List[str], **kwargs: Any) -> Any:
|
||||||
|
return type("R", (), {"returncode": 0, "stderr": ""})()
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.sys_utils.run", fake_run)
|
||||||
|
install_python_requirements(Path("/tmp/venv"), Path("/tmp/req.txt"))
|
||||||
|
|
||||||
|
def test_failure(self, monkeypatch) -> None:
|
||||||
|
def fake_run(cmd: List[str], **kwargs: Any) -> Any:
|
||||||
|
return type("R", (), {"returncode": 1, "stderr": "nope"})()
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.sys_utils.run", fake_run)
|
||||||
|
with pytest.raises(VenvCreationFailedException):
|
||||||
|
install_python_requirements(Path("/tmp/venv"), Path("/tmp/req.txt"))
|
||||||
|
|
||||||
|
|
||||||
|
class TestInstallPythonPackages:
|
||||||
|
def test_success(self, monkeypatch) -> None:
|
||||||
|
captured: List[List[str]] = []
|
||||||
|
|
||||||
|
def fake_run(cmd: List[str], **kwargs: Any) -> Any:
|
||||||
|
captured.append(cmd)
|
||||||
|
return type("R", (), {"returncode": 0, "stderr": ""})()
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.sys_utils.run", fake_run)
|
||||||
|
install_python_packages(Path("/tmp/venv"), ["a", "b"])
|
||||||
|
assert captured == [["/tmp/venv/bin/pip", "install", "a", "b"]]
|
||||||
|
|
||||||
|
|
||||||
|
class TestUpdateSystemPackageLists:
|
||||||
|
def test_skips_when_recent(self, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr("utils.sys_utils.time.time", lambda: 1000)
|
||||||
|
monkeypatch.setattr("utils.sys_utils.os.path.getmtime", lambda p: 900)
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.sys_utils.run",
|
||||||
|
lambda *a, **k: pytest.fail("should not update when recent"),
|
||||||
|
)
|
||||||
|
|
||||||
|
update_system_package_lists(silent=True)
|
||||||
|
|
||||||
|
def test_runs_when_old(self, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr("utils.sys_utils.time.time", lambda: 100_000)
|
||||||
|
monkeypatch.setattr("utils.sys_utils.os.path.getmtime", lambda p: 0)
|
||||||
|
|
||||||
|
runs: List[List[str]] = []
|
||||||
|
|
||||||
|
def fake_run(cmd: List[str], **kwargs: Any) -> Any:
|
||||||
|
runs.append(cmd)
|
||||||
|
return type("R", (), {"returncode": 0, "stderr": ""})()
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.sys_utils.run", fake_run)
|
||||||
|
update_system_package_lists(silent=True)
|
||||||
|
|
||||||
|
assert runs == [["sudo", "apt-get", "update"]]
|
||||||
|
|
||||||
|
def test_allows_releaseinfo_change(self, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr("utils.sys_utils.time.time", lambda: 100_000)
|
||||||
|
monkeypatch.setattr("utils.sys_utils.os.path.getmtime", lambda p: 0)
|
||||||
|
|
||||||
|
runs: List[List[str]] = []
|
||||||
|
|
||||||
|
def fake_run(cmd: List[str], **kwargs: Any) -> Any:
|
||||||
|
runs.append(cmd)
|
||||||
|
return type("R", (), {"returncode": 0, "stderr": ""})()
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.sys_utils.run", fake_run)
|
||||||
|
update_system_package_lists(silent=True, rls_info_change=True)
|
||||||
|
|
||||||
|
assert runs == [["sudo", "apt-get", "update", "--allow-releaseinfo-change"]]
|
||||||
|
|
||||||
|
|
||||||
|
class TestGetUpgradablePackages:
|
||||||
|
def test_parses_apt_list(self, monkeypatch) -> None:
|
||||||
|
output = (
|
||||||
|
"package1/stable 1.0 [upgradable from: 0.9]\n"
|
||||||
|
"package2/testing 2.0 [upgradable from: 1.0]\n"
|
||||||
|
)
|
||||||
|
monkeypatch.setattr("utils.sys_utils.check_output", lambda *a, **k: output)
|
||||||
|
assert get_upgradable_packages() == ["package1", "package2"]
|
||||||
|
|
||||||
|
|
||||||
|
class TestCheckPackageInstall:
|
||||||
|
def test_detects_installed(self, monkeypatch) -> None:
|
||||||
|
def fake_run(cmd: List[str], **kwargs: Any) -> Any:
|
||||||
|
return type("R", (), {"stdout": "install ok installed"})()
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.sys_utils.run", fake_run)
|
||||||
|
assert check_package_install({"git"}) == []
|
||||||
|
|
||||||
|
def test_detects_missing(self, monkeypatch) -> None:
|
||||||
|
def fake_run(cmd: List[str], **kwargs: Any) -> Any:
|
||||||
|
return type("R", (), {"stdout": "not-installed"})()
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.sys_utils.run", fake_run)
|
||||||
|
assert check_package_install({"missing"}) == ["missing"]
|
||||||
|
|
||||||
|
|
||||||
|
class TestInstallSystemPackages:
|
||||||
|
def test_runs_apt(self, monkeypatch) -> None:
|
||||||
|
runs: List[List[str]] = []
|
||||||
|
|
||||||
|
def fake_run(cmd: List[str], **kwargs: Any) -> Any:
|
||||||
|
runs.append(cmd)
|
||||||
|
return None
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.sys_utils.run", fake_run)
|
||||||
|
install_system_packages(["git", "curl"])
|
||||||
|
assert runs == [["sudo", "apt-get", "install", "-y", "git", "curl"]]
|
||||||
|
|
||||||
|
|
||||||
|
class TestUpgradeSystemPackages:
|
||||||
|
def test_runs_apt(self, monkeypatch) -> None:
|
||||||
|
runs: List[List[str]] = []
|
||||||
|
|
||||||
|
def fake_run(cmd: List[str], **kwargs: Any) -> Any:
|
||||||
|
runs.append(cmd)
|
||||||
|
return None
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.sys_utils.run", fake_run)
|
||||||
|
upgrade_system_packages(["git"])
|
||||||
|
assert runs == [["sudo", "apt-get", "upgrade", "-y", "git"]]
|
||||||
|
|
||||||
|
|
||||||
|
class TestGetIpv4Addr:
|
||||||
|
def test_returns_socket_address(self, monkeypatch) -> None:
|
||||||
|
class FakeSocket:
|
||||||
|
def __init__(self, *args):
|
||||||
|
pass
|
||||||
|
|
||||||
|
def settimeout(self, value: float) -> None:
|
||||||
|
pass
|
||||||
|
|
||||||
|
def connect(self, addr: tuple) -> None:
|
||||||
|
pass
|
||||||
|
|
||||||
|
def getsockname(self) -> tuple:
|
||||||
|
return ("192.168.1.50", 54321)
|
||||||
|
|
||||||
|
def close(self) -> None:
|
||||||
|
pass
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.sys_utils.socket.socket", FakeSocket)
|
||||||
|
assert get_ipv4_addr() == "192.168.1.50"
|
||||||
|
|
||||||
|
def test_falls_back_to_loopback(self, monkeypatch) -> None:
|
||||||
|
class FakeSocket:
|
||||||
|
def __init__(self, *args):
|
||||||
|
pass
|
||||||
|
|
||||||
|
def settimeout(self, value: float) -> None:
|
||||||
|
pass
|
||||||
|
|
||||||
|
def connect(self, addr: tuple) -> None:
|
||||||
|
raise OSError("no route")
|
||||||
|
|
||||||
|
def close(self) -> None:
|
||||||
|
pass
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.sys_utils.socket.socket", FakeSocket)
|
||||||
|
assert get_ipv4_addr() == "127.0.0.1"
|
||||||
|
|
||||||
|
|
||||||
|
class TestDownloadFile:
|
||||||
|
def test_without_progress(self, monkeypatch) -> None:
|
||||||
|
calls: List[tuple] = []
|
||||||
|
|
||||||
|
def fake_urlretrieve(url: str, target: Path, reporthook=None) -> None:
|
||||||
|
calls.append((url, str(target), reporthook))
|
||||||
|
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.sys_utils.urllib.request.urlretrieve", fake_urlretrieve
|
||||||
|
)
|
||||||
|
download_file("http://x/file", Path("/target"), show_progress=False)
|
||||||
|
assert calls == [("http://x/file", "/target", None)]
|
||||||
|
|
||||||
|
def test_with_progress(self, monkeypatch) -> None:
|
||||||
|
calls: List[tuple] = []
|
||||||
|
|
||||||
|
def fake_urlretrieve(url: str, target: Path, reporthook=None) -> None:
|
||||||
|
calls.append((url, str(target), reporthook))
|
||||||
|
if reporthook:
|
||||||
|
reporthook(1, 1024, 2048)
|
||||||
|
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.sys_utils.urllib.request.urlretrieve", fake_urlretrieve
|
||||||
|
)
|
||||||
|
download_file("http://x/file", Path("/target"), show_progress=True)
|
||||||
|
assert calls[0][2] is not None
|
||||||
|
|
||||||
|
|
||||||
|
class TestDownloadProgress:
|
||||||
|
def test_writes_to_stdout(self, capsys) -> None:
|
||||||
|
download_progress(1, 1024, 2048)
|
||||||
|
captured = capsys.readouterr()
|
||||||
|
assert "Downloading:" in captured.out
|
||||||
|
assert "50.00%" in captured.out
|
||||||
|
|
||||||
|
|
||||||
|
class TestSetNginxPermissions:
|
||||||
|
def test_no_change_when_executable(self, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.sys_utils.run",
|
||||||
|
lambda cmd, **kwargs: type("R", (), {"stdout": "drwxr-xr-x"})()
|
||||||
|
if "ls" in cmd
|
||||||
|
else pytest.fail("should not chmod"),
|
||||||
|
)
|
||||||
|
set_nginx_permissions()
|
||||||
|
|
||||||
|
def test_adds_execute(self, monkeypatch) -> None:
|
||||||
|
commands: List[Any] = []
|
||||||
|
|
||||||
|
def fake_run(cmd, **kwargs):
|
||||||
|
commands.append(cmd)
|
||||||
|
if isinstance(cmd, str):
|
||||||
|
return type("R", (), {"stdout": "drwxr------"})()
|
||||||
|
return None
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.sys_utils.run", fake_run)
|
||||||
|
monkeypatch.setattr("utils.sys_utils.Path.home", lambda: Path("/home/user"))
|
||||||
|
set_nginx_permissions()
|
||||||
|
assert ["chmod", "og+x", Path("/home/user")] in commands
|
||||||
|
|
||||||
|
|
||||||
|
class TestCmdSysctlService:
|
||||||
|
def test_runs_systemctl(self, monkeypatch) -> None:
|
||||||
|
runs: List[List[str]] = []
|
||||||
|
|
||||||
|
def fake_run(cmd: List[str], **kwargs: Any) -> Any:
|
||||||
|
runs.append(cmd)
|
||||||
|
return None
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.sys_utils.run", fake_run)
|
||||||
|
cmd_sysctl_service("klipper", "restart")
|
||||||
|
assert runs == [["sudo", "systemctl", "restart", "klipper"]]
|
||||||
|
|
||||||
|
|
||||||
|
class TestCmdSysctlManage:
|
||||||
|
def test_runs_systemctl(self, monkeypatch) -> None:
|
||||||
|
runs: List[List[str]] = []
|
||||||
|
|
||||||
|
def fake_run(cmd: List[str], **kwargs: Any) -> Any:
|
||||||
|
runs.append(cmd)
|
||||||
|
return None
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.sys_utils.run", fake_run)
|
||||||
|
cmd_sysctl_manage("daemon-reload")
|
||||||
|
assert runs == [["sudo", "systemctl", "daemon-reload"]]
|
||||||
|
|
||||||
|
|
||||||
|
class TestUnitFileExists:
|
||||||
|
def test_finds_matching_service(self, tmp_path: Path, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr("utils.sys_utils.SYSTEMD", tmp_path)
|
||||||
|
(tmp_path / "klipper.service").write_text("")
|
||||||
|
(tmp_path / "klipper-1.service").write_text("")
|
||||||
|
(tmp_path / "moonraker.service").write_text("")
|
||||||
|
|
||||||
|
assert unit_file_exists("klipper", "service") is True
|
||||||
|
assert unit_file_exists("moonraker", "service") is True
|
||||||
|
assert unit_file_exists("klipper", "timer") is False
|
||||||
|
|
||||||
|
def test_respects_exclude(self, tmp_path: Path, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr("utils.sys_utils.SYSTEMD", tmp_path)
|
||||||
|
(tmp_path / "klipper-mcu.service").write_text("")
|
||||||
|
|
||||||
|
assert unit_file_exists("klipper", "service", exclude=["mcu"]) is False
|
||||||
|
|
||||||
|
|
||||||
|
class TestLogProcess:
|
||||||
|
def test_prints_stdout(self, monkeypatch, capsys) -> None:
|
||||||
|
lines = iter(["line1\n", "line2\n", ""])
|
||||||
|
poll_results = iter([None, 0])
|
||||||
|
|
||||||
|
class FakeStdout:
|
||||||
|
def fileno(self) -> int:
|
||||||
|
return 7
|
||||||
|
|
||||||
|
def readline(self) -> str:
|
||||||
|
return next(lines)
|
||||||
|
|
||||||
|
class FakeProcess:
|
||||||
|
stdout = FakeStdout()
|
||||||
|
|
||||||
|
def poll(self):
|
||||||
|
return next(poll_results)
|
||||||
|
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.sys_utils.select.select", lambda r, w, x: ([7], [], [])
|
||||||
|
)
|
||||||
|
log_process(FakeProcess()) # type: ignore[arg-type]
|
||||||
|
|
||||||
|
captured = capsys.readouterr()
|
||||||
|
assert "line1" in captured.out
|
||||||
|
assert "line2" in captured.out
|
||||||
|
|
||||||
|
|
||||||
|
class TestCreateServiceFile:
|
||||||
|
def test_writes_via_tee(self, monkeypatch) -> None:
|
||||||
|
runs: List[tuple] = []
|
||||||
|
|
||||||
|
def fake_run(cmd: List[str], **kwargs: Any) -> Any:
|
||||||
|
runs.append((cmd, kwargs.get("input")))
|
||||||
|
return None
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.sys_utils.SYSTEMD", Path("/etc/systemd/system"))
|
||||||
|
monkeypatch.setattr("utils.sys_utils.run", fake_run)
|
||||||
|
create_service_file("klipper.service", "[Unit]\n")
|
||||||
|
|
||||||
|
assert runs[0][0] == [
|
||||||
|
"sudo",
|
||||||
|
"tee",
|
||||||
|
Path("/etc/systemd/system/klipper.service"),
|
||||||
|
]
|
||||||
|
assert runs[0][1] == b"[Unit]\n"
|
||||||
|
|
||||||
|
|
||||||
|
class TestCreateEnvFile:
|
||||||
|
def test_writes_file(self, tmp_path: Path) -> None:
|
||||||
|
path = tmp_path / "env"
|
||||||
|
create_env_file(path, "KEY=value\n")
|
||||||
|
assert path.read_text() == "KEY=value\n"
|
||||||
|
|
||||||
|
|
||||||
|
class TestRemoveSystemService:
|
||||||
|
def test_rejects_bad_name(self) -> None:
|
||||||
|
with pytest.raises(ValueError):
|
||||||
|
remove_system_service("klipper")
|
||||||
|
|
||||||
|
def test_skips_missing_file(self, tmp_path: Path, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr("utils.sys_utils.SYSTEMD", tmp_path)
|
||||||
|
remove_system_service("klipper.service")
|
||||||
|
|
||||||
|
def test_full_removal(self, monkeypatch) -> None:
|
||||||
|
sysd = Path("/fake/systemd")
|
||||||
|
service_file = sysd / "klipper.service"
|
||||||
|
monkeypatch.setattr("utils.sys_utils.SYSTEMD", sysd)
|
||||||
|
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.sys_utils.Path.exists",
|
||||||
|
lambda self: str(self) == str(service_file),
|
||||||
|
)
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.sys_utils.Path.is_file",
|
||||||
|
lambda self: str(self) == str(service_file),
|
||||||
|
)
|
||||||
|
|
||||||
|
service_calls: List[tuple] = []
|
||||||
|
manage_calls: List[str] = []
|
||||||
|
removed: List[Path] = []
|
||||||
|
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.sys_utils.cmd_sysctl_service",
|
||||||
|
lambda name, action: service_calls.append((name, action)),
|
||||||
|
)
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.sys_utils.cmd_sysctl_manage",
|
||||||
|
lambda action: manage_calls.append(action),
|
||||||
|
)
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.sys_utils.remove_with_sudo", lambda p: removed.append(p)
|
||||||
|
)
|
||||||
|
|
||||||
|
remove_system_service("klipper.service")
|
||||||
|
|
||||||
|
assert service_calls == [
|
||||||
|
("klipper.service", "stop"),
|
||||||
|
("klipper.service", "disable"),
|
||||||
|
]
|
||||||
|
assert removed == [service_file]
|
||||||
|
assert manage_calls == ["daemon-reload", "reset-failed"]
|
||||||
|
|
||||||
|
|
||||||
|
class _FakeInstanceType:
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
_FakeInstanceType.__name__ = "Klipper"
|
||||||
|
|
||||||
|
|
||||||
|
class TestGetServiceFilePath:
|
||||||
|
def test_builds_path(self, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr("utils.sys_utils.SYSTEMD", Path("/etc/systemd/system"))
|
||||||
|
assert get_service_file_path(_FakeInstanceType, "") == Path(
|
||||||
|
"/etc/systemd/system/klipper.service"
|
||||||
|
)
|
||||||
|
assert get_service_file_path(_FakeInstanceType, "1") == Path(
|
||||||
|
"/etc/systemd/system/klipper-1.service"
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class TestGetDistroInfo:
|
||||||
|
def test_parses_os_release(self, monkeypatch) -> None:
|
||||||
|
content = """
|
||||||
|
ID="ubuntu"
|
||||||
|
ID_LIKE="debian"
|
||||||
|
VERSION_ID="22.04"
|
||||||
|
"""
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.sys_utils.check_output", lambda *a, **k: content.encode()
|
||||||
|
)
|
||||||
|
assert get_distro_info() == ("ubuntu", "22.04")
|
||||||
|
|
||||||
|
def test_remaps_raspbian(self, monkeypatch) -> None:
|
||||||
|
content = """
|
||||||
|
ID="raspbian"
|
||||||
|
ID_LIKE="debian"
|
||||||
|
VERSION_ID="11"
|
||||||
|
"""
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.sys_utils.check_output", lambda *a, **k: content.encode()
|
||||||
|
)
|
||||||
|
assert get_distro_info() == ("debian", "11")
|
||||||
|
|
||||||
|
def test_raises_on_missing_id(self, monkeypatch) -> None:
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.sys_utils.check_output", lambda *a, **k: b'VERSION_ID="1"\n'
|
||||||
|
)
|
||||||
|
with pytest.raises(ValueError):
|
||||||
|
get_distro_info()
|
||||||
|
|
||||||
|
|
||||||
|
class TestGetSystemTimezone:
|
||||||
|
def test_from_etc_timezone(self, monkeypatch) -> None:
|
||||||
|
def fake_open(path: str, mode: str = "r", *args, **kwargs):
|
||||||
|
if path == "/etc/timezone":
|
||||||
|
return StringIO("Europe/Berlin\n")
|
||||||
|
return builtins.open(path, mode, *args, **kwargs)
|
||||||
|
|
||||||
|
monkeypatch.setattr("builtins.open", fake_open)
|
||||||
|
assert get_system_timezone() == "Europe/Berlin"
|
||||||
|
|
||||||
|
def test_fallback_to_timedatectl(self, monkeypatch) -> None:
|
||||||
|
def fake_open(path: str, mode: str = "r", *args, **kwargs):
|
||||||
|
raise FileNotFoundError(path)
|
||||||
|
|
||||||
|
monkeypatch.setattr("builtins.open", fake_open)
|
||||||
|
|
||||||
|
def fake_run(cmd: List[str], **kwargs: Any) -> Any:
|
||||||
|
class Result:
|
||||||
|
stdout = "Timezone=America/New_York\n"
|
||||||
|
|
||||||
|
return Result()
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.sys_utils.run", fake_run)
|
||||||
|
assert get_system_timezone() == "America/New_York"
|
||||||
|
|
||||||
|
def test_fallback_to_readlink(self, monkeypatch) -> None:
|
||||||
|
def fake_open(path: str, mode: str = "r", *args, **kwargs):
|
||||||
|
raise FileNotFoundError(path)
|
||||||
|
|
||||||
|
monkeypatch.setattr("builtins.open", fake_open)
|
||||||
|
|
||||||
|
def fake_run(cmd: List[str], **kwargs: Any) -> Any:
|
||||||
|
if cmd[:2] == ["timedatectl", "show"]:
|
||||||
|
raise CalledProcessError(1, "timedatectl")
|
||||||
|
|
||||||
|
class Result:
|
||||||
|
stdout = "/usr/share/zoneinfo/Asia/Tokyo\n"
|
||||||
|
|
||||||
|
return Result()
|
||||||
|
|
||||||
|
monkeypatch.setattr("utils.sys_utils.run", fake_run)
|
||||||
|
assert get_system_timezone() == "Asia/Tokyo"
|
||||||
|
|
||||||
|
def test_defaults_to_utc(self, monkeypatch) -> None:
|
||||||
|
def fake_open(path: str, mode: str = "r", *args, **kwargs):
|
||||||
|
raise FileNotFoundError(path)
|
||||||
|
|
||||||
|
monkeypatch.setattr("builtins.open", fake_open)
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"utils.sys_utils.run",
|
||||||
|
lambda *a, **k: (_ for _ in ()).throw(CalledProcessError(1, "timedatectl")),
|
||||||
|
)
|
||||||
|
assert get_system_timezone() == "UTC"
|
||||||
+1
-1
@@ -33,5 +33,5 @@ warn_unreachable = true
|
|||||||
|
|
||||||
[tool.pytest.ini_options]
|
[tool.pytest.ini_options]
|
||||||
minversion = "8.2.1"
|
minversion = "8.2.1"
|
||||||
testpaths = ["kiauh/core/simple_config_parser/tests"]
|
testpaths = ["kiauh/core/simple_config_parser/tests", "kiauh/utils/tests"]
|
||||||
pythonpath = ["kiauh"]
|
pythonpath = ["kiauh"]
|
||||||
|
|||||||
Reference in New Issue
Block a user