Files
takopi/tests/test_telegram_client.py
T

233 lines
6.6 KiB
Python

import httpx
import pytest
from takopi.logging import setup_logging
from takopi.telegram.client import TelegramClient, TelegramRetryAfter
from takopi.telegram.client_api import HttpBotClient
@pytest.mark.anyio
async def test_telegram_429_no_retry() -> None:
calls: list[int] = []
def handler(request: httpx.Request) -> httpx.Response:
calls.append(1)
return httpx.Response(
429,
json={
"ok": False,
"description": "retry",
"parameters": {"retry_after": 3},
},
request=request,
)
transport = httpx.MockTransport(handler)
client = httpx.AsyncClient(transport=transport)
try:
api = HttpBotClient("123:abcDEF_ghij", http_client=client)
with pytest.raises(TelegramRetryAfter) as exc:
await api._post("sendMessage", {"chat_id": 1, "text": "hi"})
finally:
await client.aclose()
assert exc.value.retry_after == 3
assert len(calls) == 1
@pytest.mark.anyio
async def test_no_token_in_logs_on_http_error(
capsys: pytest.CaptureFixture[str],
) -> None:
token = "123:abcDEF_ghij"
setup_logging(debug=True)
def handler(request: httpx.Request) -> httpx.Response:
return httpx.Response(500, text="oops", request=request)
transport = httpx.MockTransport(handler)
client = httpx.AsyncClient(transport=transport)
try:
api = HttpBotClient(token, http_client=client)
await api._post("getUpdates", {"timeout": 1})
finally:
await client.aclose()
out = capsys.readouterr().out
assert token not in out
assert "bot[REDACTED]" in out
@pytest.mark.anyio
async def test_telegram_429_no_retry_post_form() -> None:
calls: list[int] = []
def handler(request: httpx.Request) -> httpx.Response:
calls.append(1)
return httpx.Response(
429,
json={
"ok": False,
"description": "retry",
"parameters": {"retry_after": 2},
},
request=request,
)
transport = httpx.MockTransport(handler)
client = httpx.AsyncClient(transport=transport)
try:
api = HttpBotClient("123:abcDEF_ghij", http_client=client)
with pytest.raises(TelegramRetryAfter) as exc:
await api._post_form(
"sendDocument",
{"chat_id": 1},
files={"document": ("note.txt", b"hi")},
)
finally:
await client.aclose()
assert exc.value.retry_after == 2
assert len(calls) == 1
@pytest.mark.anyio
async def test_telegram_429_defaults_retry_after_on_bad_body() -> None:
def handler(request: httpx.Request) -> httpx.Response:
return httpx.Response(429, text="nope", request=request)
transport = httpx.MockTransport(handler)
client = httpx.AsyncClient(transport=transport)
try:
api = HttpBotClient("123:abcDEF_ghij", http_client=client)
with pytest.raises(TelegramRetryAfter) as exc:
await api._post("sendMessage", {"chat_id": 1, "text": "hi"})
finally:
await client.aclose()
assert exc.value.retry_after == 5.0
@pytest.mark.anyio
async def test_telegram_ok_false_returns_none() -> None:
def handler(request: httpx.Request) -> httpx.Response:
return httpx.Response(
200,
json={"ok": False, "error_code": 400, "description": "bad"},
request=request,
)
transport = httpx.MockTransport(handler)
client = httpx.AsyncClient(transport=transport)
try:
api = HttpBotClient("123:abcDEF_ghij", http_client=client)
result = await api._post("getUpdates", {"timeout": 1})
finally:
await client.aclose()
assert result is None
@pytest.mark.anyio
async def test_telegram_invalid_payload_returns_none() -> None:
def handler(request: httpx.Request) -> httpx.Response:
return httpx.Response(200, json=["not", "a", "dict"], request=request)
transport = httpx.MockTransport(handler)
client = httpx.AsyncClient(transport=transport)
try:
api = HttpBotClient("123:abcDEF_ghij", http_client=client)
result = await api._post("getUpdates", {"timeout": 1})
finally:
await client.aclose()
assert result is None
@pytest.mark.anyio
async def test_telegram_decode_failure_returns_none() -> None:
def handler(request: httpx.Request) -> httpx.Response:
return httpx.Response(
200,
json={"ok": True, "result": {"username": "bot-only"}},
request=request,
)
transport = httpx.MockTransport(handler)
client = httpx.AsyncClient(transport=transport)
try:
tg = TelegramClient("123:abcDEF_ghij", http_client=client)
result = await tg.get_me()
finally:
await client.aclose()
assert result is None
@pytest.mark.anyio
async def test_telegram_download_file_retries_on_429() -> None:
calls: list[int] = []
sleeps: list[float] = []
async def sleep(delay: float) -> None:
sleeps.append(delay)
def handler(request: httpx.Request) -> httpx.Response:
calls.append(1)
if len(calls) == 1:
return httpx.Response(
429,
json={"ok": False, "parameters": {"retry_after": 3}},
request=request,
)
return httpx.Response(200, content=b"ok", request=request)
transport = httpx.MockTransport(handler)
client = httpx.AsyncClient(transport=transport)
try:
tg = TelegramClient("123:abcDEF_ghij", http_client=client, sleep=sleep)
payload = await tg.download_file("path/to/file")
finally:
await client.aclose()
assert payload == b"ok"
assert sleeps == [3.0]
assert len(calls) == 2
@pytest.mark.anyio
async def test_telegram_download_file_429_defaults_retry_after_on_bad_body() -> None:
sleeps: list[float] = []
async def sleep(delay: float) -> None:
sleeps.append(delay)
calls: list[int] = []
def handler(request: httpx.Request) -> httpx.Response:
calls.append(1)
if len(calls) == 1:
return httpx.Response(429, text="nope", request=request)
return httpx.Response(200, content=b"ok", request=request)
transport = httpx.MockTransport(handler)
client = httpx.AsyncClient(transport=transport)
try:
tg = TelegramClient("123:abcDEF_ghij", http_client=client, sleep=sleep)
payload = await tg.download_file("path")
finally:
await client.aclose()
assert payload == b"ok"
assert sleeps == [5.0]
assert len(calls) == 2