mirror of
https://github.com/Alishahryar1/free-claude-code.git
synced 2026-06-02 06:13:46 +02:00
fbb1d6586d
Route these providers through POST /messages with vendor headers and bases (including Kimi model list on OpenAI /v1/models). Remove Z.ai from OpenAI-chat server-tool rejection; extend tests and README.
108 lines
3.0 KiB
Python
108 lines
3.0 KiB
Python
"""Tests for Z.ai native Anthropic Messages provider."""
|
|
|
|
from contextlib import asynccontextmanager
|
|
from unittest.mock import AsyncMock, patch
|
|
|
|
import pytest
|
|
|
|
from api.models.anthropic import Message, MessagesRequest
|
|
from config.constants import ANTHROPIC_DEFAULT_MAX_OUTPUT_TOKENS
|
|
from providers.base import ProviderConfig
|
|
from providers.defaults import ZAI_DEFAULT_BASE
|
|
from providers.exceptions import InvalidRequestError
|
|
from providers.zai import ZaiProvider
|
|
|
|
|
|
@pytest.fixture
|
|
def zai_config():
|
|
return ProviderConfig(
|
|
api_key="test_zai_key",
|
|
base_url=ZAI_DEFAULT_BASE,
|
|
rate_limit=10,
|
|
rate_window=60,
|
|
enable_thinking=True,
|
|
)
|
|
|
|
|
|
@pytest.fixture(autouse=True)
|
|
def mock_rate_limiter():
|
|
@asynccontextmanager
|
|
async def _slot():
|
|
yield
|
|
|
|
with patch("providers.anthropic_messages.GlobalRateLimiter") as mock:
|
|
instance = mock.get_scoped_instance.return_value
|
|
|
|
async def _passthrough(fn, *args, **kwargs):
|
|
return await fn(*args, **kwargs)
|
|
|
|
instance.execute_with_retry = AsyncMock(side_effect=_passthrough)
|
|
instance.concurrency_slot.side_effect = _slot
|
|
yield instance
|
|
|
|
|
|
@pytest.fixture
|
|
def zai_provider(zai_config):
|
|
return ZaiProvider(zai_config)
|
|
|
|
|
|
def test_init(zai_config):
|
|
with patch("httpx.AsyncClient") as mock_client:
|
|
provider = ZaiProvider(zai_config)
|
|
assert provider._api_key == "test_zai_key"
|
|
assert provider._base_url == ZAI_DEFAULT_BASE
|
|
assert mock_client.called
|
|
|
|
|
|
def test_request_headers(zai_provider):
|
|
h = zai_provider._request_headers()
|
|
assert h["x-api-key"] == "test_zai_key"
|
|
assert h["anthropic-version"] == "2023-06-01"
|
|
|
|
|
|
def test_model_list_headers(zai_provider):
|
|
h = zai_provider._model_list_headers()
|
|
assert h["x-api-key"] == "test_zai_key"
|
|
|
|
|
|
def test_build_request_body_native(zai_provider):
|
|
request = MessagesRequest(
|
|
model="glm-5.1",
|
|
max_tokens=100,
|
|
messages=[Message(role="user", content="Hello")],
|
|
)
|
|
body = zai_provider._build_request_body(request)
|
|
assert body["model"] == "glm-5.1"
|
|
assert body["stream"] is True
|
|
assert body["max_tokens"] == 100
|
|
|
|
|
|
def test_build_request_body_default_max_tokens(zai_provider):
|
|
request = MessagesRequest(
|
|
model="m",
|
|
messages=[Message(role="user", content="x")],
|
|
)
|
|
body = zai_provider._build_request_body(request)
|
|
assert body["max_tokens"] == ANTHROPIC_DEFAULT_MAX_OUTPUT_TOKENS
|
|
|
|
|
|
def test_build_request_body_rejects_extra_body(zai_provider):
|
|
request = MessagesRequest.model_validate(
|
|
{
|
|
"model": "m",
|
|
"messages": [{"role": "user", "content": "x"}],
|
|
"extra_body": {"x": 1},
|
|
}
|
|
)
|
|
with pytest.raises(InvalidRequestError, match="does not support extra_body"):
|
|
zai_provider._build_request_body(request)
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_cleanup_aclose(zai_provider):
|
|
zai_provider._client = AsyncMock()
|
|
|
|
await zai_provider.cleanup()
|
|
|
|
zai_provider._client.aclose.assert_awaited_once()
|