mirror of
https://github.com/kuhyx/testsAndMisc.git
synced 2026-07-04 20:23:11 +02:00
Fix ruff violations in ~15 source files and ~60+ test files to minimize per-file-ignores in pyproject.toml. Remaining ignores are justified with comments explaining why each suppression is necessary. Source fixes: FBT003 (keyword args), S310 (URL validation), SLF001 (private access), T201 (print→logging), C901 (complexity), E501 (line length), E402 (import order). Test fixes: SIM117 (combined with), FBT (boolean args), PERF203 (try in loop), S310/S607 (URLs/executables), E402/E501 (imports/lines), S108 (tmp paths), PLR0913 (too many args), ARG (unused args), ANN (type annotations), RUF059 (unused unpacked vars), PT019 (fixture naming). Remaining per-file-ignores (with justifications): - Tests: ARG, D, PLC0415, PLR2004, S101, SLF001 - music_gen sources: PLC0415 (heavy ML lazy imports) - moviepy_showcase: PLC0415 (circular dependency) - generate_images: PLR0913 (matplotlib helpers need many params) - praca_magisterska_video: E501, E402 (long paths, mpl.use)
491 lines
16 KiB
Python
491 lines
16 KiB
Python
"""Tests for UI transitions, timer logic, and workout detail screens."""
|
|
|
|
from __future__ import annotations
|
|
|
|
import tkinter as tk
|
|
from typing import TYPE_CHECKING
|
|
from unittest.mock import MagicMock
|
|
|
|
from python_pkg.screen_locker.screen_lock import (
|
|
SUBMIT_DELAY_DEMO,
|
|
SUBMIT_DELAY_PRODUCTION,
|
|
)
|
|
from python_pkg.screen_locker.tests.conftest import create_locker
|
|
|
|
if TYPE_CHECKING:
|
|
from pathlib import Path
|
|
|
|
_TK_TCLERROR = tk.TclError
|
|
|
|
|
|
class TestUITransitions:
|
|
"""Tests for UI state transitions."""
|
|
|
|
def test_clear_container(
|
|
self,
|
|
mock_tk: MagicMock,
|
|
mock_sys_exit: MagicMock,
|
|
tmp_path: Path,
|
|
) -> None:
|
|
"""Test clear_container destroys all child widgets."""
|
|
locker = create_locker(mock_tk, tmp_path)
|
|
|
|
# Set up mock children
|
|
mock_child1 = MagicMock()
|
|
mock_child2 = MagicMock()
|
|
locker.container.winfo_children.return_value = [
|
|
mock_child1,
|
|
mock_child2,
|
|
]
|
|
|
|
locker.clear_container()
|
|
|
|
mock_child1.destroy.assert_called_once()
|
|
mock_child2.destroy.assert_called_once()
|
|
|
|
def test_unlock_screen_saves_and_schedules_close(
|
|
self,
|
|
mock_tk: MagicMock,
|
|
mock_sys_exit: MagicMock,
|
|
tmp_path: Path,
|
|
) -> None:
|
|
"""Test unlock_screen saves log and schedules close."""
|
|
locker = create_locker(mock_tk, tmp_path)
|
|
locker.log_file = tmp_path / "workout_log.json"
|
|
locker.workout_data = {"type": "running"}
|
|
|
|
locker.unlock_screen()
|
|
|
|
# Check that after() was called to schedule close
|
|
locker.root.after.assert_called()
|
|
|
|
def test_lockout_starts_countdown(
|
|
self,
|
|
mock_tk: MagicMock,
|
|
mock_sys_exit: MagicMock,
|
|
tmp_path: Path,
|
|
) -> None:
|
|
"""Test lockout initializes countdown timer."""
|
|
locker = create_locker(mock_tk, tmp_path)
|
|
|
|
locker.lockout()
|
|
|
|
# lockout() sets remaining_time to lockout_time (10 in demo mode)
|
|
# then calls update_lockout_countdown() which decrements it by 1
|
|
assert locker.remaining_time == 9 # 10 - 1 after first update
|
|
|
|
def test_close_destroys_root_and_exits(
|
|
self,
|
|
mock_tk: MagicMock,
|
|
mock_sys_exit: MagicMock,
|
|
tmp_path: Path,
|
|
) -> None:
|
|
"""Test close destroys root window and exits."""
|
|
locker = create_locker(mock_tk, tmp_path)
|
|
|
|
locker.close()
|
|
|
|
locker.root.destroy.assert_called_once()
|
|
mock_sys_exit.assert_called_with(0)
|
|
|
|
|
|
class TestTimerLogic:
|
|
"""Tests for timer countdown logic."""
|
|
|
|
def test_update_lockout_countdown_decrements(
|
|
self,
|
|
mock_tk: MagicMock,
|
|
mock_sys_exit: MagicMock,
|
|
tmp_path: Path,
|
|
) -> None:
|
|
"""Test countdown decrements remaining time."""
|
|
locker = create_locker(mock_tk, tmp_path)
|
|
locker.remaining_time = 5
|
|
locker.countdown_label = MagicMock()
|
|
|
|
locker.update_lockout_countdown()
|
|
|
|
assert locker.remaining_time == 4
|
|
locker.root.after.assert_called_with(1000, locker.update_lockout_countdown)
|
|
|
|
def test_update_lockout_countdown_at_zero(
|
|
self,
|
|
mock_tk: MagicMock,
|
|
mock_sys_exit: MagicMock,
|
|
tmp_path: Path,
|
|
) -> None:
|
|
"""Test countdown at zero returns to workout question."""
|
|
locker = create_locker(mock_tk, tmp_path)
|
|
locker.remaining_time = 0
|
|
locker.countdown_label = MagicMock()
|
|
object.__setattr__(locker, "ask_workout_done", MagicMock())
|
|
|
|
locker.update_lockout_countdown()
|
|
|
|
locker.ask_workout_done.assert_called_once()
|
|
|
|
def test_update_submit_timer_countdown(
|
|
self,
|
|
mock_tk: MagicMock,
|
|
mock_sys_exit: MagicMock,
|
|
tmp_path: Path,
|
|
) -> None:
|
|
"""Test submit timer counts down."""
|
|
locker = create_locker(mock_tk, tmp_path)
|
|
locker.submit_unlock_time = 5
|
|
locker.timer_label = MagicMock()
|
|
locker.submit_btn = MagicMock()
|
|
locker.entries_to_check = []
|
|
|
|
locker.update_submit_timer()
|
|
|
|
assert locker.submit_unlock_time == 4
|
|
locker.root.after.assert_called()
|
|
|
|
def test_update_submit_timer_enables_when_filled(
|
|
self,
|
|
mock_tk: MagicMock,
|
|
mock_sys_exit: MagicMock,
|
|
tmp_path: Path,
|
|
) -> None:
|
|
"""Test submit enabled when timer done and entries filled."""
|
|
locker = create_locker(mock_tk, tmp_path)
|
|
locker.submit_unlock_time = 0
|
|
locker.timer_label = MagicMock()
|
|
locker.submit_btn = MagicMock()
|
|
mock_entry = MagicMock()
|
|
mock_entry.get.return_value = "some value"
|
|
locker.entries_to_check = [mock_entry]
|
|
locker.submit_command = MagicMock()
|
|
|
|
locker.update_submit_timer()
|
|
|
|
locker.submit_btn.config.assert_called()
|
|
|
|
def test_update_submit_timer_waits_for_entries(
|
|
self,
|
|
mock_tk: MagicMock,
|
|
mock_sys_exit: MagicMock,
|
|
tmp_path: Path,
|
|
) -> None:
|
|
"""Test submit waits when entries not filled."""
|
|
locker = create_locker(mock_tk, tmp_path)
|
|
locker.submit_unlock_time = 0
|
|
locker.timer_label = MagicMock()
|
|
locker.submit_btn = MagicMock()
|
|
mock_entry = MagicMock()
|
|
mock_entry.get.return_value = "" # Empty entry
|
|
locker.entries_to_check = [mock_entry]
|
|
|
|
locker.update_submit_timer()
|
|
|
|
locker.root.after.assert_called_with(1000, locker.check_entries_filled)
|
|
|
|
def test_update_submit_timer_handles_tcl_error(
|
|
self,
|
|
mock_tk: MagicMock,
|
|
mock_sys_exit: MagicMock,
|
|
tmp_path: Path,
|
|
) -> None:
|
|
"""Test timer handles TclError when widgets destroyed."""
|
|
locker = create_locker(mock_tk, tmp_path)
|
|
locker.submit_unlock_time = 5
|
|
locker.timer_label = MagicMock()
|
|
locker.timer_label.config.side_effect = _TK_TCLERROR("widget destroyed")
|
|
|
|
# Should not raise
|
|
locker.update_submit_timer()
|
|
|
|
def test_check_entries_filled_enables_submit(
|
|
self,
|
|
mock_tk: MagicMock,
|
|
mock_sys_exit: MagicMock,
|
|
tmp_path: Path,
|
|
) -> None:
|
|
"""Test check_entries_filled enables submit when all filled."""
|
|
locker = create_locker(mock_tk, tmp_path)
|
|
locker.timer_label = MagicMock()
|
|
locker.submit_btn = MagicMock()
|
|
mock_entry = MagicMock()
|
|
mock_entry.get.return_value = "value"
|
|
locker.entries_to_check = [mock_entry]
|
|
locker.submit_command = MagicMock()
|
|
|
|
locker.check_entries_filled()
|
|
|
|
locker.submit_btn.config.assert_called()
|
|
|
|
def test_check_entries_filled_continues_waiting(
|
|
self,
|
|
mock_tk: MagicMock,
|
|
mock_sys_exit: MagicMock,
|
|
tmp_path: Path,
|
|
) -> None:
|
|
"""Test check_entries_filled continues waiting when not filled."""
|
|
locker = create_locker(mock_tk, tmp_path)
|
|
locker.timer_label = MagicMock()
|
|
locker.submit_btn = MagicMock()
|
|
mock_entry = MagicMock()
|
|
mock_entry.get.return_value = ""
|
|
locker.entries_to_check = [mock_entry]
|
|
|
|
locker.check_entries_filled()
|
|
|
|
locker.root.after.assert_called_with(1000, locker.check_entries_filled)
|
|
|
|
def test_check_entries_filled_handles_tcl_error(
|
|
self,
|
|
mock_tk: MagicMock,
|
|
mock_sys_exit: MagicMock,
|
|
tmp_path: Path,
|
|
) -> None:
|
|
"""Test check_entries_filled handles TclError."""
|
|
locker = create_locker(mock_tk, tmp_path)
|
|
locker.timer_label = MagicMock()
|
|
mock_entry = MagicMock()
|
|
mock_entry.get.side_effect = _TK_TCLERROR("widget destroyed")
|
|
locker.entries_to_check = [mock_entry]
|
|
|
|
# Should not raise
|
|
locker.check_entries_filled()
|
|
|
|
|
|
class TestAskWorkoutType:
|
|
"""Tests for ask_workout_type method."""
|
|
|
|
def test_ask_workout_type_creates_buttons(
|
|
self,
|
|
mock_tk: MagicMock,
|
|
mock_sys_exit: MagicMock,
|
|
tmp_path: Path,
|
|
) -> None:
|
|
"""Test ask_workout_type creates running and strength buttons."""
|
|
locker = create_locker(mock_tk, tmp_path)
|
|
object.__setattr__(locker, "clear_container", MagicMock())
|
|
|
|
locker.ask_workout_type()
|
|
|
|
locker.clear_container.assert_called_once()
|
|
# Verify Label and Button were called
|
|
mock_tk.Label.assert_called()
|
|
mock_tk.Button.assert_called()
|
|
|
|
|
|
class TestAskRunningDetails:
|
|
"""Tests for ask_running_details method."""
|
|
|
|
def test_ask_running_details_sets_workout_type(
|
|
self,
|
|
mock_tk: MagicMock,
|
|
mock_sys_exit: MagicMock,
|
|
tmp_path: Path,
|
|
) -> None:
|
|
"""Test ask_running_details sets workout type to running."""
|
|
locker = create_locker(mock_tk, tmp_path)
|
|
object.__setattr__(locker, "clear_container", MagicMock())
|
|
object.__setattr__(locker, "update_submit_timer", MagicMock())
|
|
|
|
locker.ask_running_details()
|
|
|
|
assert locker.workout_data["type"] == "running"
|
|
locker.clear_container.assert_called_once()
|
|
|
|
def test_ask_running_details_creates_entry_fields(
|
|
self,
|
|
mock_tk: MagicMock,
|
|
mock_sys_exit: MagicMock,
|
|
tmp_path: Path,
|
|
) -> None:
|
|
"""Test ask_running_details creates entry fields."""
|
|
locker = create_locker(mock_tk, tmp_path)
|
|
object.__setattr__(locker, "clear_container", MagicMock())
|
|
object.__setattr__(locker, "update_submit_timer", MagicMock())
|
|
|
|
locker.ask_running_details()
|
|
|
|
# Verify Entry fields were created
|
|
mock_tk.Entry.assert_called()
|
|
assert hasattr(locker, "distance_entry")
|
|
assert hasattr(locker, "time_entry")
|
|
assert hasattr(locker, "pace_entry")
|
|
|
|
def test_ask_running_details_sets_timer(
|
|
self,
|
|
mock_tk: MagicMock,
|
|
mock_sys_exit: MagicMock,
|
|
tmp_path: Path,
|
|
) -> None:
|
|
"""Test ask_running_details initializes submit timer."""
|
|
locker = create_locker(mock_tk, tmp_path)
|
|
object.__setattr__(locker, "clear_container", MagicMock())
|
|
object.__setattr__(locker, "update_submit_timer", MagicMock())
|
|
|
|
locker.ask_running_details()
|
|
|
|
assert locker.submit_unlock_time == SUBMIT_DELAY_DEMO
|
|
locker.update_submit_timer.assert_called_once()
|
|
|
|
|
|
class TestAskStrengthDetails:
|
|
"""Tests for ask_strength_details method."""
|
|
|
|
def test_ask_strength_details_sets_workout_type(
|
|
self,
|
|
mock_tk: MagicMock,
|
|
mock_sys_exit: MagicMock,
|
|
tmp_path: Path,
|
|
) -> None:
|
|
"""Test ask_strength_details sets workout type to strength."""
|
|
locker = create_locker(mock_tk, tmp_path)
|
|
object.__setattr__(locker, "clear_container", MagicMock())
|
|
object.__setattr__(locker, "update_submit_timer", MagicMock())
|
|
|
|
locker.ask_strength_details()
|
|
|
|
assert locker.workout_data["type"] == "strength"
|
|
locker.clear_container.assert_called_once()
|
|
|
|
def test_ask_strength_details_creates_entry_fields(
|
|
self,
|
|
mock_tk: MagicMock,
|
|
mock_sys_exit: MagicMock,
|
|
tmp_path: Path,
|
|
) -> None:
|
|
"""Test ask_strength_details creates entry fields."""
|
|
locker = create_locker(mock_tk, tmp_path)
|
|
object.__setattr__(locker, "clear_container", MagicMock())
|
|
object.__setattr__(locker, "update_submit_timer", MagicMock())
|
|
|
|
locker.ask_strength_details()
|
|
|
|
# Verify Entry fields were created
|
|
mock_tk.Entry.assert_called()
|
|
assert hasattr(locker, "exercises_entry")
|
|
assert hasattr(locker, "sets_entry")
|
|
assert hasattr(locker, "reps_entry")
|
|
assert hasattr(locker, "weights_entry")
|
|
assert hasattr(locker, "total_weight_entry")
|
|
|
|
def test_ask_strength_details_sets_timer(
|
|
self,
|
|
mock_tk: MagicMock,
|
|
mock_sys_exit: MagicMock,
|
|
tmp_path: Path,
|
|
) -> None:
|
|
"""Test ask_strength_details initializes submit timer."""
|
|
locker = create_locker(mock_tk, tmp_path)
|
|
object.__setattr__(locker, "clear_container", MagicMock())
|
|
object.__setattr__(locker, "update_submit_timer", MagicMock())
|
|
|
|
locker.ask_strength_details()
|
|
|
|
assert locker.submit_unlock_time == SUBMIT_DELAY_DEMO
|
|
locker.update_submit_timer.assert_called_once()
|
|
|
|
def test_ask_strength_details_production_timer(
|
|
self,
|
|
mock_tk: MagicMock,
|
|
mock_sys_exit: MagicMock,
|
|
tmp_path: Path,
|
|
) -> None:
|
|
"""Test production mode uses longer submit delay."""
|
|
locker = create_locker(mock_tk, tmp_path, demo_mode=False)
|
|
object.__setattr__(locker, "clear_container", MagicMock())
|
|
object.__setattr__(locker, "update_submit_timer", MagicMock())
|
|
|
|
locker.ask_strength_details()
|
|
|
|
assert locker.submit_unlock_time == SUBMIT_DELAY_PRODUCTION
|
|
|
|
|
|
class TestAskWorkoutDone:
|
|
"""Tests for ask_workout_done method."""
|
|
|
|
def test_ask_workout_done_creates_buttons(
|
|
self,
|
|
mock_tk: MagicMock,
|
|
mock_sys_exit: MagicMock,
|
|
tmp_path: Path,
|
|
) -> None:
|
|
"""Test ask_workout_done creates yes/no buttons."""
|
|
locker = create_locker(mock_tk, tmp_path)
|
|
object.__setattr__(locker, "clear_container", MagicMock())
|
|
|
|
locker.ask_workout_done()
|
|
|
|
locker.clear_container.assert_called_once()
|
|
mock_tk.Label.assert_called()
|
|
mock_tk.Button.assert_called()
|
|
|
|
|
|
class TestAskIfSick:
|
|
"""Tests for ask_if_sick method."""
|
|
|
|
def test_ask_if_sick_displays_dialog(
|
|
self, mock_tk: MagicMock, mock_sys_exit: MagicMock, tmp_path: Path
|
|
) -> None:
|
|
"""Test ask_if_sick shows sick day question."""
|
|
locker = create_locker(mock_tk, tmp_path)
|
|
object.__setattr__(locker, "clear_container", MagicMock())
|
|
locker.ask_if_sick()
|
|
locker.clear_container.assert_called_once()
|
|
mock_tk.Label.assert_called()
|
|
|
|
|
|
class TestSickQuestionButtons:
|
|
"""Tests for _sick_question_buttons method."""
|
|
|
|
def test_creates_buttons(
|
|
self, mock_tk: MagicMock, mock_sys_exit: MagicMock, tmp_path: Path
|
|
) -> None:
|
|
"""Test _sick_question_buttons creates yes/no buttons."""
|
|
locker = create_locker(mock_tk, tmp_path)
|
|
locker._sick_question_buttons()
|
|
mock_tk.Button.assert_called()
|
|
|
|
|
|
class TestGetSickDayStatus:
|
|
"""Tests for _get_sick_day_status method."""
|
|
|
|
def test_already_adjusted_today(
|
|
self, mock_tk: MagicMock, mock_sys_exit: MagicMock, tmp_path: Path
|
|
) -> None:
|
|
"""Test status when sick mode already used today."""
|
|
locker = create_locker(mock_tk, tmp_path)
|
|
object.__setattr__(
|
|
locker, "_sick_mode_used_today", MagicMock(return_value=True)
|
|
)
|
|
text, color = locker._get_sick_day_status()
|
|
assert "already adjusted" in text
|
|
assert color == "#ffaa00"
|
|
|
|
def test_adjustment_success(
|
|
self, mock_tk: MagicMock, mock_sys_exit: MagicMock, tmp_path: Path
|
|
) -> None:
|
|
"""Test status when shutdown time adjusted successfully."""
|
|
locker = create_locker(mock_tk, tmp_path)
|
|
object.__setattr__(
|
|
locker, "_sick_mode_used_today", MagicMock(return_value=False)
|
|
)
|
|
object.__setattr__(
|
|
locker, "_adjust_shutdown_time_earlier", MagicMock(return_value=True)
|
|
)
|
|
text, color = locker._get_sick_day_status()
|
|
assert "earlier" in text
|
|
assert color == "#00aa00"
|
|
|
|
def test_adjustment_failure(
|
|
self, mock_tk: MagicMock, mock_sys_exit: MagicMock, tmp_path: Path
|
|
) -> None:
|
|
"""Test status when adjustment fails."""
|
|
locker = create_locker(mock_tk, tmp_path)
|
|
object.__setattr__(
|
|
locker, "_sick_mode_used_today", MagicMock(return_value=False)
|
|
)
|
|
object.__setattr__(
|
|
locker, "_adjust_shutdown_time_earlier", MagicMock(return_value=False)
|
|
)
|
|
text, color = locker._get_sick_day_status()
|
|
assert "Could not adjust" in text
|
|
assert color == "#ff4444"
|