hermes-agent/tests/gateway/test_telegram_progress_edit_transient.py
kshitijk4poor 66827f8947 chore: prune unused imports and duplicate import redefinitions
Remove unused imports (F401) and duplicate/shadowed import
redefinitions (F811) across the codebase using ruff's safe
autofixes. No behavioral changes -- imports only.

- ~1400 safe autofixes applied across 644 files (net -1072 lines)
- __init__.py re-exports preserved (excluded from F401 removal so
  public re-export surfaces stay intact)
- Re-exports that are imported or monkeypatched by tests but look
  unused in their defining module are kept with explicit # noqa:
  F401 (gateway/run.py load_dotenv; run_agent re-exports from
  agent.message_sanitization, agent.context_compressor,
  agent.retry_utils, agent.prompt_builder, agent.process_bootstrap,
  agent.codex_responses_adapter)
- Unsafe F841 (unused-variable) fixes deliberately skipped -- those
  can change behavior when the RHS has side effects
- ruff lints remain disabled in pyproject.toml (only PLW1514 is
  selected); this is a one-time cleanup, not a config change

Verification:
- python -m compileall: clean
- pytest --collect-only: all 27161 tests collect (zero import errors)
- core entry points import clean (run_agent, model_tools, cli,
  toolsets, hermes_state, batch_runner, gateway)
- static scan: every name any test imports directly from an edited
  module still resolves
2026-05-28 22:26:25 -07:00

181 lines
5.9 KiB
Python

"""Tests for transient-error handling in Telegram progress-message editing.
Issue: #27828
When ``edit_message_text`` fails with a transient network error (e.g.
``httpx.ConnectError``), the gateway must NOT permanently disable progress-
message editing. Only permanent failures (flood control, message-not-found,
permissions) should set ``can_edit = False``.
Two layers are tested:
1. The ``_TRANSIENT_EDIT_MARKERS`` / retryable classification logic in
``TelegramAdapter.edit_message``.
2. The ``send_progress_messages`` caller in ``run.py`` honours
``result.retryable`` and keeps ``can_edit = True``.
"""
from __future__ import annotations
import pytest
from gateway.platforms.base import SendResult
# ---------------------------------------------------------------------------
# Helpers
# ---------------------------------------------------------------------------
_TRANSIENT_MARKERS = (
"connecterror",
"connect error",
"connection error",
"networkerror",
"network error",
"timed out",
"readtimeout",
"writetimeout",
"server disconnected",
"temporarily unavailable",
"temporary failure",
"httpx",
)
_PERMANENT_MARKERS = (
"message to edit not found",
"message can't be edited",
"not enough rights",
"message_id_invalid",
)
def _is_transient(error_str: str) -> bool:
"""Mirrors the classification logic added to TelegramAdapter.edit_message."""
err = error_str.lower()
return any(m in err for m in _TRANSIENT_MARKERS)
def _is_permanent(error_str: str) -> bool:
err = error_str.lower()
return any(m in err for m in _PERMANENT_MARKERS)
# ---------------------------------------------------------------------------
# 1. Error classification — transient vs permanent
# ---------------------------------------------------------------------------
@pytest.mark.parametrize("error_str", [
"httpx.ConnectError: Connection refused",
"telegram.error.NetworkError: httpx.ConnectError",
"NetworkError: remote end closed connection without response",
"httpx.ReadTimeout: read timed out",
"ReadTimeout: timed out",
"Server disconnected",
"Temporarily unavailable",
"Temporary failure in name resolution",
"Connection error: failed to connect",
])
def test_transient_errors_are_classified_as_transient(error_str):
"""Network / transient errors must be classified as retryable."""
assert _is_transient(error_str), (
f"Expected {error_str!r} to be transient"
)
@pytest.mark.parametrize("error_str", [
"Bad Request: message to edit not found",
"Bad Request: message can't be edited",
"Bad Request: not enough rights to edit the message",
"Bad Request: MESSAGE_ID_INVALID",
"flood_control:30.0",
"Forbidden: bot was blocked by the user",
])
def test_permanent_errors_are_not_transient(error_str):
"""Permanent edit failures must NOT be classified as retryable."""
assert not _is_transient(error_str), (
f"Expected {error_str!r} to be permanent (non-transient)"
)
# ---------------------------------------------------------------------------
# 2. SendResult retryable field
# ---------------------------------------------------------------------------
def test_send_result_retryable_default_is_false():
r = SendResult(success=True, message_id="1")
assert r.retryable is False
def test_send_result_retryable_can_be_set_true():
r = SendResult(success=False, error="httpx.ConnectError: ...", retryable=True)
assert r.retryable is True
def test_send_result_retryable_false_for_permanent():
r = SendResult(success=False, error="message to edit not found")
assert r.retryable is False
# ---------------------------------------------------------------------------
# 3. run.py logic — retryable result must NOT set can_edit=False
# We simulate the relevant block from send_progress_messages():
#
# if not result.success:
# if getattr(result, 'retryable', False):
# continue # <-- keep can_edit=True
# ...
# can_edit = False
#
# ---------------------------------------------------------------------------
def _simulate_progress_loop(edit_results):
"""
Simulate the can_edit decision for a sequence of edit_message results.
Returns the final value of can_edit after processing all results.
"""
can_edit = True
for result in edit_results:
if not result.success:
if getattr(result, "retryable", False):
# Transient — keep can_edit True and skip to next cycle
continue
can_edit = False
break
return can_edit
def test_transient_failure_keeps_can_edit_true():
"""A single transient network error must not disable progress editing."""
results = [
SendResult(success=False, error="httpx.ConnectError", retryable=True),
SendResult(success=True, message_id="42"),
]
assert _simulate_progress_loop(results) is True
def test_permanent_failure_sets_can_edit_false():
"""A permanent edit failure must disable progress editing."""
results = [
SendResult(success=False, error="message to edit not found", retryable=False),
]
assert _simulate_progress_loop(results) is False
def test_multiple_transient_then_success_keeps_can_edit_true():
"""Multiple transient failures followed by success keep can_edit=True."""
results = [
SendResult(success=False, error="httpx.ConnectError", retryable=True),
SendResult(success=False, error="server disconnected", retryable=True),
SendResult(success=True, message_id="99"),
]
assert _simulate_progress_loop(results) is True
def test_flood_control_sets_can_edit_false():
"""Flood control (non-retryable) must disable progress editing."""
results = [
SendResult(success=False, error="flood_control:30.0", retryable=False),
]
assert _simulate_progress_loop(results) is False