test_crawapi_http.py 2.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081
  1. """V3-M1A: shared crawapi HTTP base unit tests."""
  2. from __future__ import annotations
  3. import httpx
  4. import pytest
  5. from content_agent.errors import ContentAgentError, ErrorCode
  6. from content_agent.integrations.crawapi_http import (
  7. RateLimiter,
  8. is_rate_limit_business_error,
  9. post_crawapi_json,
  10. )
  11. class FakeHttpClient:
  12. def __init__(self, responses):
  13. self.responses = list(responses)
  14. self.requests = []
  15. def post(self, url, json, headers, timeout):
  16. self.requests.append({"url": url, "json": json})
  17. response = self.responses.pop(0)
  18. if isinstance(response, Exception):
  19. raise response
  20. return response
  21. def _response(status_code, data):
  22. return httpx.Response(
  23. status_code, json=data, request=httpx.Request("POST", "http://crawapi.test/x")
  24. )
  25. def _post(responses, **kwargs):
  26. return post_crawapi_json(
  27. http_client=FakeHttpClient(responses),
  28. base_url="http://crawapi.test/",
  29. path="x",
  30. payload={},
  31. operation="probe",
  32. timeout_seconds=60.0,
  33. business_codes=kwargs.get("business_codes", set()),
  34. rate_limiter=kwargs.get("rate_limiter"),
  35. rate_limit_bucket=kwargs.get("rate_limit_bucket"),
  36. )
  37. def test_rate_limiter_waits_min_interval_between_same_bucket():
  38. clock = {"now": 0.0}
  39. sleeps: list[float] = []
  40. limiter = RateLimiter(
  41. min_interval_seconds=12.0,
  42. now_fn=lambda: clock["now"],
  43. sleep_fn=lambda s: (sleeps.append(s), clock.__setitem__("now", clock["now"] + s)),
  44. )
  45. limiter.wait("b")
  46. limiter.wait("b")
  47. assert sleeps == [12.0]
  48. def test_http_429_maps_to_platform_rate_limited():
  49. with pytest.raises(ContentAgentError) as exc:
  50. _post([_response(429, {"msg": "slow down"})])
  51. assert exc.value.error_code == ErrorCode.PLATFORM_RATE_LIMITED
  52. def test_message_token_maps_to_platform_rate_limited():
  53. with pytest.raises(ContentAgentError) as exc:
  54. _post([_response(200, {"code": 50000, "msg": "请求频繁"})])
  55. assert exc.value.error_code == ErrorCode.PLATFORM_RATE_LIMITED
  56. def test_bad_response_non_dict_raises_runtime_error():
  57. with pytest.raises(RuntimeError, match="bad_response"):
  58. _post([_response(200, ["not", "a", "dict"])])
  59. def test_business_codes_param_classifies_rate_limit():
  60. assert is_rate_limit_business_error("30005", {}, business_codes={"30005"}) is True
  61. assert is_rate_limit_business_error("30005", {}, business_codes=set()) is False