mirror of
https://github.com/NVIDIA-NeMo/DataDesigner
synced 2026-05-24 09:48:29 +00:00
Preserves tree from previous docs-website head: 5e47d33ea8. This branch is a CI-managed publish artifact like gh-pages; source provenance is tracked in commit messages rather than Git ancestry.
228 lines
9.4 KiB
Python
228 lines
9.4 KiB
Python
# SPDX-FileCopyrightText: Copyright (c) 2025-2026 NVIDIA CORPORATION & AFFILIATES. All rights reserved.
|
|
# SPDX-License-Identifier: Apache-2.0
|
|
|
|
import warnings
|
|
|
|
import pytest
|
|
|
|
from data_designer.config.mcp import LocalStdioMCPProvider
|
|
from data_designer.engine.errors import NoModelProvidersError
|
|
from data_designer.engine.model_provider import (
|
|
MCPProviderRegistry,
|
|
ModelProvider,
|
|
ModelProviderRegistry,
|
|
UnknownProviderError,
|
|
resolve_model_provider_registry,
|
|
)
|
|
|
|
|
|
@pytest.fixture
|
|
def stub_foo_provider():
|
|
return ModelProvider(name="foo", endpoint="https://foo.com", provider_type="foo")
|
|
|
|
|
|
@pytest.fixture
|
|
def stub_bar_provider():
|
|
return ModelProvider(name="bar", endpoint="https://bar.com", provider_type="bar")
|
|
|
|
|
|
def test_must_have_at_least_one_provider():
|
|
with pytest.raises(ValueError):
|
|
ModelProviderRegistry(providers=[], default="a")
|
|
|
|
with pytest.raises(ValueError):
|
|
ModelProviderRegistry(providers=[])
|
|
|
|
|
|
def test_defined_default_must_exist(stub_foo_provider: ModelProvider):
|
|
with pytest.raises(ValueError):
|
|
ModelProviderRegistry(providers=[stub_foo_provider], default="bar")
|
|
|
|
|
|
def test_multiple_providers_requires_explicit_default(
|
|
stub_foo_provider: ModelProvider, stub_bar_provider: ModelProvider
|
|
):
|
|
with pytest.raises(ValueError):
|
|
ModelProviderRegistry(providers=[stub_foo_provider, stub_bar_provider])
|
|
|
|
|
|
def test_implicit_default(stub_foo_provider: ModelProvider):
|
|
registry = ModelProviderRegistry(providers=[stub_foo_provider])
|
|
|
|
assert registry.get_provider(None) == stub_foo_provider
|
|
|
|
|
|
def test_no_duplicate_provider_names(stub_foo_provider: ModelProvider):
|
|
with pytest.raises(ValueError):
|
|
ModelProviderRegistry(providers=[stub_foo_provider, stub_foo_provider], default="foo")
|
|
|
|
|
|
def test_get_provider(stub_foo_provider: ModelProvider, stub_bar_provider: ModelProvider):
|
|
# Multi-provider construction with an explicit default exercises the #589
|
|
# deprecation path; wrap so this test stays green if the project ever runs
|
|
# with ``-W error::DeprecationWarning``.
|
|
with pytest.warns(DeprecationWarning, match="ModelProviderRegistry.default is deprecated"):
|
|
registry = ModelProviderRegistry(
|
|
providers=[stub_foo_provider, stub_bar_provider],
|
|
default="foo",
|
|
)
|
|
|
|
assert registry.get_provider(None) == stub_foo_provider
|
|
assert registry.get_provider("foo") == stub_foo_provider
|
|
assert registry.get_provider("bar") == stub_bar_provider
|
|
|
|
with pytest.raises(UnknownProviderError):
|
|
registry.get_provider("quux")
|
|
|
|
|
|
def test_resolve_model_provider_registry(stub_foo_provider: ModelProvider) -> None:
|
|
"""Test resolve_model_provider_registry creates a registry from providers."""
|
|
registry = resolve_model_provider_registry([stub_foo_provider])
|
|
|
|
assert len(registry.providers) == 1
|
|
assert registry.get_default_provider_name() == "foo"
|
|
|
|
|
|
def test_resolve_model_provider_registry_with_explicit_default(
|
|
stub_foo_provider: ModelProvider, stub_bar_provider: ModelProvider
|
|
) -> None:
|
|
"""Test resolve_model_provider_registry with explicit default.
|
|
|
|
The multi-provider/explicit-default path is the deprecated one (see #589),
|
|
so the construction emits a ``DeprecationWarning``. Wrap the call in
|
|
``pytest.warns`` so this test stays green if the project ever runs under
|
|
``-W error::DeprecationWarning``.
|
|
"""
|
|
with pytest.warns(DeprecationWarning, match="ModelProviderRegistry.default is deprecated"):
|
|
registry = resolve_model_provider_registry([stub_foo_provider, stub_bar_provider], default_provider_name="bar")
|
|
|
|
assert registry.get_default_provider_name() == "bar"
|
|
|
|
|
|
def test_resolve_model_provider_registry_empty_error() -> None:
|
|
"""Test resolve_model_provider_registry raises error for empty providers."""
|
|
with pytest.raises(NoModelProvidersError, match="At least one model provider"):
|
|
resolve_model_provider_registry([])
|
|
|
|
|
|
def test_explicit_default_emits_deprecation_warning(stub_foo_provider: ModelProvider) -> None:
|
|
"""Regression for #589: passing ``default=`` explicitly to ``ModelProviderRegistry``
|
|
must emit a ``DeprecationWarning``. The registry-level default field is on its
|
|
way out; users should specify ``provider=`` per ``ModelConfig`` instead.
|
|
"""
|
|
with pytest.warns(DeprecationWarning, match="ModelProviderRegistry.default is deprecated"):
|
|
ModelProviderRegistry(providers=[stub_foo_provider], default="foo")
|
|
|
|
|
|
def test_no_default_does_not_emit_deprecation_warning(stub_foo_provider: ModelProvider) -> None:
|
|
"""Pin the post-deprecation happy path: omitting ``default=`` (single-provider
|
|
case) must NOT emit a warning, since callers haven't opted into the deprecated
|
|
field.
|
|
"""
|
|
with warnings.catch_warnings():
|
|
warnings.simplefilter("error", DeprecationWarning)
|
|
ModelProviderRegistry(providers=[stub_foo_provider])
|
|
|
|
|
|
def test_explicit_default_none_does_not_emit_deprecation_warning(stub_foo_provider: ModelProvider) -> None:
|
|
"""Pin the predicate tightening from PR #594 review: passing ``default=None``
|
|
explicitly is semantically equivalent to omitting it (caller is opting *out*
|
|
of a registry-level default), so the deprecation must NOT fire.
|
|
"""
|
|
with warnings.catch_warnings():
|
|
warnings.simplefilter("error", DeprecationWarning)
|
|
ModelProviderRegistry(providers=[stub_foo_provider], default=None)
|
|
|
|
|
|
def test_explicit_default_warning_attributes_to_user_frame(
|
|
stub_foo_provider: ModelProvider, stub_bar_provider: ModelProvider
|
|
) -> None:
|
|
"""Regression for PR #594 review (andreatgretel): the ``default=`` deprecation
|
|
warning must attribute to the *user's* call site, not the pydantic-internal
|
|
or ``data_designer`` library frame that emits it. Library-attributed
|
|
``DeprecationWarning`` entries are silenced under Python's default
|
|
``ignore::DeprecationWarning`` filter, so attribution determines whether
|
|
the warning is actually visible.
|
|
|
|
Construction goes through ``resolve_model_provider_registry`` so the walk
|
|
has to escape both pydantic (validator dispatch) and ``data_designer``
|
|
(the helper that builds the registry) before landing on the test frame.
|
|
"""
|
|
with warnings.catch_warnings(record=True) as caught:
|
|
warnings.simplefilter("always", DeprecationWarning)
|
|
resolve_model_provider_registry([stub_foo_provider, stub_bar_provider], default_provider_name="bar")
|
|
|
|
matches = [w for w in caught if "ModelProviderRegistry.default is deprecated" in str(w.message)]
|
|
assert len(matches) == 1, [str(w.message) for w in caught]
|
|
assert matches[0].filename == __file__, (
|
|
f"Warning attributed to {matches[0].filename!r} (line {matches[0].lineno}) "
|
|
f"instead of the test file. Library-attributed DeprecationWarnings are "
|
|
f"silenced under default filters."
|
|
)
|
|
|
|
|
|
def test_resolve_single_provider_quiet_under_deprecation(stub_foo_provider: ModelProvider) -> None:
|
|
"""Pin the q3 tweak: ``resolve_model_provider_registry`` skips ``default=``
|
|
in the single-provider case so common construction paths stay quiet under
|
|
the #589 deprecation warning.
|
|
"""
|
|
with warnings.catch_warnings():
|
|
warnings.simplefilter("error", DeprecationWarning)
|
|
registry = resolve_model_provider_registry([stub_foo_provider])
|
|
|
|
assert registry.get_default_provider_name() == "foo"
|
|
|
|
|
|
def test_resolve_multi_provider_emits_deprecation_warning(
|
|
stub_foo_provider: ModelProvider, stub_bar_provider: ModelProvider
|
|
) -> None:
|
|
"""Multi-provider registries currently require ``default``, so
|
|
``resolve_model_provider_registry`` keeps passing it. That construction
|
|
path is the deprecated one users should migrate off; the warning fires
|
|
accordingly.
|
|
"""
|
|
with pytest.warns(DeprecationWarning, match="ModelProviderRegistry.default is deprecated"):
|
|
resolve_model_provider_registry([stub_foo_provider, stub_bar_provider])
|
|
|
|
|
|
def test_mcp_provider_registry_empty() -> None:
|
|
"""Test MCPProviderRegistry can be created empty."""
|
|
registry = MCPProviderRegistry()
|
|
|
|
assert len(registry.providers) == 0
|
|
|
|
|
|
def test_mcp_provider_registry_with_providers() -> None:
|
|
"""Test MCPProviderRegistry with providers."""
|
|
provider = LocalStdioMCPProvider(name="test-provider", command="test-cmd")
|
|
registry = MCPProviderRegistry(providers=[provider])
|
|
|
|
assert len(registry.providers) == 1
|
|
assert registry.get_provider("test-provider") == provider
|
|
|
|
|
|
def test_mcp_provider_registry_duplicate_names() -> None:
|
|
"""Test MCPProviderRegistry raises error for duplicate provider names."""
|
|
provider1 = LocalStdioMCPProvider(name="test-provider", command="test-cmd")
|
|
provider2 = LocalStdioMCPProvider(name="test-provider", command="test-cmd-2")
|
|
|
|
with pytest.raises(ValueError, match="duplicate"):
|
|
MCPProviderRegistry(providers=[provider1, provider2])
|
|
|
|
|
|
def test_mcp_provider_registry_unknown_provider() -> None:
|
|
"""Test MCPProviderRegistry raises error for unknown provider."""
|
|
registry = MCPProviderRegistry()
|
|
|
|
with pytest.raises(UnknownProviderError, match="registered"):
|
|
registry.get_provider("unknown-provider")
|
|
|
|
|
|
def test_mcp_provider_registry_is_empty() -> None:
|
|
"""Test MCPProviderRegistry is_empty method."""
|
|
empty_registry = MCPProviderRegistry()
|
|
assert empty_registry.is_empty() is True
|
|
|
|
provider = LocalStdioMCPProvider(name="test-provider", command="test-cmd")
|
|
registry_with_providers = MCPProviderRegistry(providers=[provider])
|
|
assert registry_with_providers.is_empty() is False
|