Files
free-claude-code/smoke/prereq/test_provider_prereq_live.py
Alishahryar1 f96f541c0a fix(smoke): accept reasoning-only streams and add text placeholder
- OpenAI-compat: emit minimal text block when only reasoning_content streams
  (e.g. NIM) so clients get a text segment.
- Provider prereq: pass if text or thinking content is non-empty after strip.
- Add unit test for reasoning-only stream placeholder text.
2026-04-26 12:55:18 -07:00

113 lines
3.8 KiB
Python

from __future__ import annotations
import time
import httpx
import pytest
from core.anthropic.stream_contracts import (
assert_anthropic_stream_contract,
text_content,
thinking_content,
)
from smoke.lib.config import SmokeConfig, auth_headers
from smoke.lib.e2e import ProviderMatrixDriver
from smoke.lib.http import collect_message_stream, message_payload
from smoke.lib.server import start_server
from smoke.lib.skips import (
skip_if_upstream_unavailable_events,
skip_if_upstream_unavailable_exception,
)
pytestmark = [pytest.mark.live, pytest.mark.smoke_target("providers")]
def test_model_mapping_configuration_is_consistent(smoke_config: SmokeConfig) -> None:
models = smoke_config.provider_models()
if not models:
pytest.skip("no configured provider models with usable credentials/base URLs")
for provider_model in models:
assert "/" in provider_model.full_model
assert provider_model.model_name
def test_mixed_provider_model_mapping_when_configured(
smoke_config: SmokeConfig,
) -> None:
models = smoke_config.provider_models()
providers = {provider_model.provider for provider_model in models}
if len(providers) < 2:
pytest.skip("configure MODEL_* with at least two provider prefixes")
sources = {provider_model.source for provider_model in models}
assert sources <= {"MODEL", "MODEL_OPUS", "MODEL_SONNET", "MODEL_HAIKU"}
assert len(providers) >= 2
def test_configured_provider_models_stream_successfully(
smoke_config: SmokeConfig,
) -> None:
models = ProviderMatrixDriver(smoke_config).provider_smoke_models()
failures: list[str] = []
for provider_model in models:
try:
with start_server(
smoke_config,
env_overrides={
"MODEL": provider_model.full_model,
"MESSAGING_PLATFORM": "none",
},
name=f"provider-{provider_model.provider}",
) as server:
events = collect_message_stream(
server,
message_payload(smoke_config.prompt, model="fcc-smoke-default"),
smoke_config,
)
skip_if_upstream_unavailable_events(events)
assert_anthropic_stream_contract(events)
has_text = bool(text_content(events).strip())
has_thinking = bool(thinking_content(events).strip())
assert has_text or has_thinking, (
"provider returned no visible text or thinking content"
)
except Exception as exc:
skip_if_upstream_unavailable_exception(exc)
failures.append(
f"{provider_model.source}={provider_model.full_model}: "
f"{type(exc).__name__}: {exc}"
)
assert not failures, "\n".join(failures)
@pytest.mark.smoke_target("rate_limit")
def test_client_disconnect_mid_stream_does_not_crash_server(
smoke_config: SmokeConfig,
) -> None:
provider_model = ProviderMatrixDriver(smoke_config).first_model()
with start_server(
smoke_config,
env_overrides={
"MODEL": provider_model.full_model,
"MESSAGING_PLATFORM": "none",
},
name="disconnect",
) as server:
with httpx.stream(
"POST",
f"{server.base_url}/v1/messages",
headers=auth_headers(),
json=message_payload(smoke_config.prompt, model="fcc-smoke-default"),
timeout=smoke_config.timeout_s,
) as response:
assert response.status_code == 200, response.read()
for _line in response.iter_lines():
break
time.sleep(0.5)
health = httpx.get(f"{server.base_url}/health", timeout=5)
assert health.status_code == 200