mirror of
https://github.com/microsoft/agent-framework.git
synced 2026-06-16 21:04:09 +08:00
9c094573e8
* Further support for declarative python workflows * Add tests. Clean up for typing and formatting * Improvements and cleanup * Typing cleanup. Improve docstrings * Proper code in docstrings * Fix malformed code-block directive in docstring * Remove dead links * PR feedback * Address PR feedback * Address PR feedback * Remove sl * Update devui frontend * More cleanup * Fix uv lock * Skip Py 3.14 tests as powerfx doesn't support it * Fix mypy error * Fix for tool calls * Removed stale docstring * Fix lint * Standardize on .NET namespaces. Revert DevUI changes (bring in later) * Implement remaining items for Python declarative support to match dotnet
695 lines
20 KiB
Python
695 lines
20 KiB
Python
# Copyright (c) Microsoft. All rights reserved.
|
|
|
|
import sys
|
|
from pathlib import Path
|
|
from typing import Any
|
|
|
|
import pytest
|
|
import yaml
|
|
|
|
from agent_framework_declarative._models import (
|
|
AgentDefinition,
|
|
AgentManifest,
|
|
AnonymousConnection,
|
|
ApiKeyConnection,
|
|
ArrayProperty,
|
|
CodeInterpreterTool,
|
|
Connection,
|
|
CustomTool,
|
|
FileSearchTool,
|
|
FunctionTool,
|
|
McpServerApprovalMode,
|
|
McpServerToolAlwaysRequireApprovalMode,
|
|
McpServerToolNeverRequireApprovalMode,
|
|
McpServerToolSpecifyApprovalMode,
|
|
McpTool,
|
|
ModelResource,
|
|
ObjectProperty,
|
|
OpenApiTool,
|
|
PromptAgent,
|
|
Property,
|
|
PropertySchema,
|
|
ReferenceConnection,
|
|
RemoteConnection,
|
|
Resource,
|
|
ToolResource,
|
|
WebSearchTool,
|
|
agent_schema_dispatch,
|
|
)
|
|
|
|
pytestmark = pytest.mark.skipif(sys.version_info >= (3, 14), reason="Skipping on Python 3.14+")
|
|
|
|
try:
|
|
import powerfx # noqa: F401
|
|
|
|
_powerfx_available = True
|
|
except (ImportError, RuntimeError):
|
|
_powerfx_available = False
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"yaml_content,expected_type,expected_attributes",
|
|
[
|
|
# Agent Manifest (no kind field)
|
|
(
|
|
"""
|
|
name: my-manifest
|
|
description: A test manifest
|
|
""",
|
|
AgentManifest,
|
|
{"name": "my-manifest", "description": "A test manifest"},
|
|
),
|
|
# PromptAgent
|
|
(
|
|
"""
|
|
kind: Prompt
|
|
name: assistant
|
|
description: A helpful assistant
|
|
model:
|
|
id: gpt-4
|
|
""",
|
|
PromptAgent,
|
|
{"name": "assistant", "description": "A helpful assistant"},
|
|
),
|
|
# AgentDefinition
|
|
(
|
|
"""
|
|
kind: Agent
|
|
name: base-agent
|
|
description: A base agent
|
|
""",
|
|
AgentDefinition,
|
|
{"name": "base-agent", "description": "A base agent"},
|
|
),
|
|
# ModelResource
|
|
(
|
|
"""
|
|
kind: Model
|
|
name: my-model
|
|
id: gpt-4
|
|
""",
|
|
ModelResource,
|
|
{"name": "my-model", "id": "gpt-4"},
|
|
),
|
|
# ToolResource
|
|
(
|
|
"""
|
|
kind: Tool
|
|
name: my-tool
|
|
id: search-tool
|
|
""",
|
|
ToolResource,
|
|
{"name": "my-tool", "id": "search-tool"},
|
|
),
|
|
# Resource (base)
|
|
(
|
|
"""
|
|
kind: Resource
|
|
name: generic-resource
|
|
""",
|
|
Resource,
|
|
{"name": "generic-resource"},
|
|
),
|
|
# FunctionTool
|
|
(
|
|
"""
|
|
kind: function
|
|
name: get_weather
|
|
description: Get the weather
|
|
""",
|
|
FunctionTool,
|
|
{"name": "get_weather", "description": "Get the weather"},
|
|
),
|
|
# CustomTool
|
|
(
|
|
"""
|
|
kind: custom
|
|
name: custom_tool
|
|
description: A custom tool
|
|
""",
|
|
CustomTool,
|
|
{"name": "custom_tool", "description": "A custom tool"},
|
|
),
|
|
# WebSearchTool
|
|
(
|
|
"""
|
|
kind: web_search
|
|
name: search
|
|
description: Search the web
|
|
""",
|
|
WebSearchTool,
|
|
{"name": "search", "description": "Search the web"},
|
|
),
|
|
# FileSearchTool
|
|
(
|
|
"""
|
|
kind: file_search
|
|
name: file_search
|
|
description: Search files
|
|
""",
|
|
FileSearchTool,
|
|
{"name": "file_search", "description": "Search files"},
|
|
),
|
|
# McpTool
|
|
(
|
|
"""
|
|
kind: mcp
|
|
name: mcp_tool
|
|
description: An MCP tool
|
|
serverName: my-server
|
|
""",
|
|
McpTool,
|
|
{"name": "mcp_tool", "serverName": "my-server"},
|
|
),
|
|
# OpenApiTool
|
|
(
|
|
"""
|
|
kind: openapi
|
|
name: api_tool
|
|
description: An OpenAPI tool
|
|
specification: https://api.example.com/openapi.json
|
|
""",
|
|
OpenApiTool,
|
|
{"name": "api_tool", "specification": "https://api.example.com/openapi.json"},
|
|
),
|
|
# CodeInterpreterTool
|
|
(
|
|
"""
|
|
kind: code_interpreter
|
|
name: code_tool
|
|
description: A code interpreter tool
|
|
""",
|
|
CodeInterpreterTool,
|
|
{"name": "code_tool", "description": "A code interpreter tool"},
|
|
),
|
|
# ReferenceConnection
|
|
(
|
|
"""
|
|
kind: reference
|
|
name: my-connection
|
|
target: target-connection
|
|
""",
|
|
ReferenceConnection,
|
|
{"name": "my-connection", "target": "target-connection"},
|
|
),
|
|
# RemoteConnection
|
|
(
|
|
"""
|
|
kind: remote
|
|
endpoint: https://api.example.com
|
|
""",
|
|
RemoteConnection,
|
|
{"endpoint": "https://api.example.com"},
|
|
),
|
|
# ApiKeyConnection
|
|
(
|
|
"""
|
|
kind: key
|
|
apiKey: secret-key
|
|
endpoint: https://api.example.com
|
|
""",
|
|
ApiKeyConnection,
|
|
{"apiKey": "secret-key", "endpoint": "https://api.example.com"},
|
|
),
|
|
# AnonymousConnection
|
|
(
|
|
"""
|
|
kind: anonymous
|
|
endpoint: https://api.example.com
|
|
""",
|
|
AnonymousConnection,
|
|
{"endpoint": "https://api.example.com"},
|
|
),
|
|
# Connection (base)
|
|
(
|
|
"""
|
|
kind: connection
|
|
authenticationMode: oauth
|
|
""",
|
|
Connection,
|
|
{"authenticationMode": "oauth"},
|
|
),
|
|
# ArrayProperty
|
|
(
|
|
"""
|
|
kind: array
|
|
name: items
|
|
description: An array of items
|
|
""",
|
|
ArrayProperty,
|
|
{"name": "items", "description": "An array of items"},
|
|
),
|
|
# ObjectProperty
|
|
(
|
|
"""
|
|
kind: object
|
|
name: config
|
|
description: Configuration object
|
|
""",
|
|
ObjectProperty,
|
|
{"name": "config", "description": "Configuration object"},
|
|
),
|
|
# Property (base)
|
|
(
|
|
"""
|
|
kind: property
|
|
name: field
|
|
description: A property field
|
|
""",
|
|
Property,
|
|
{"name": "field", "description": "A property field"},
|
|
),
|
|
# McpServerToolAlwaysRequireApprovalMode
|
|
(
|
|
"""
|
|
kind: always
|
|
""",
|
|
McpServerToolAlwaysRequireApprovalMode,
|
|
{},
|
|
),
|
|
# McpServerToolNeverRequireApprovalMode
|
|
(
|
|
"""
|
|
kind: never
|
|
""",
|
|
McpServerToolNeverRequireApprovalMode,
|
|
{},
|
|
),
|
|
# McpServerToolSpecifyApprovalMode
|
|
(
|
|
"""
|
|
kind: specify
|
|
alwaysRequireApprovalTools: []
|
|
neverRequireApprovalTools: []
|
|
""",
|
|
McpServerToolSpecifyApprovalMode,
|
|
{},
|
|
),
|
|
# McpServerApprovalMode (base)
|
|
(
|
|
"""
|
|
kind: approval_mode
|
|
""",
|
|
McpServerApprovalMode,
|
|
{},
|
|
),
|
|
],
|
|
)
|
|
def test_agent_schema_dispatch_all_types(yaml_content: str, expected_type: type, expected_attributes: dict[str, Any]):
|
|
"""Test that agent_schema_dispatch correctly loads all MAML object types."""
|
|
result = agent_schema_dispatch(yaml.safe_load(yaml_content))
|
|
|
|
# Check the type is correct
|
|
assert isinstance(result, expected_type), f"Expected {expected_type.__name__}, got {type(result).__name__}"
|
|
|
|
# Check expected attributes
|
|
for attr_name, attr_value in expected_attributes.items():
|
|
assert hasattr(result, attr_name), f"Result missing attribute '{attr_name}'"
|
|
assert getattr(result, attr_name) == attr_value, (
|
|
f"Attribute '{attr_name}' has value {getattr(result, attr_name)}, expected {attr_value}"
|
|
)
|
|
|
|
|
|
def test_agent_schema_dispatch_unknown_kind():
|
|
"""Test that agent_schema_dispatch returns None for unknown kind."""
|
|
yaml_content = """
|
|
kind: unknown_type
|
|
name: test
|
|
"""
|
|
result = agent_schema_dispatch(yaml.safe_load(yaml_content))
|
|
assert result is None
|
|
|
|
|
|
def test_agent_schema_dispatch_complex_agent_manifest():
|
|
"""Test loading a complex agent manifest with nested objects."""
|
|
yaml_content = """
|
|
name: complex-manifest
|
|
description: A complete manifest
|
|
template:
|
|
kind: Prompt
|
|
name: assistant
|
|
description: A helpful assistant
|
|
model:
|
|
id: gpt-4
|
|
provider: openai
|
|
tools:
|
|
- kind: web_search
|
|
name: search
|
|
description: Search the web
|
|
- kind: function
|
|
name: calculator
|
|
description: Calculate math
|
|
resources:
|
|
- kind: model
|
|
name: model1
|
|
id: gpt-4
|
|
- kind: tool
|
|
name: tool1
|
|
id: search
|
|
"""
|
|
result = agent_schema_dispatch(yaml.safe_load(yaml_content))
|
|
|
|
assert isinstance(result, AgentManifest)
|
|
assert result.name == "complex-manifest"
|
|
assert result.description == "A complete manifest"
|
|
assert isinstance(result.template, PromptAgent)
|
|
assert result.template.name == "assistant"
|
|
assert len(result.resources) == 2
|
|
assert isinstance(result.resources[0], ModelResource)
|
|
assert isinstance(result.resources[1], ToolResource)
|
|
|
|
|
|
def test_agent_schema_dispatch_prompt_agent_with_tools():
|
|
"""Test loading a prompt agent with multiple tools."""
|
|
yaml_content = """
|
|
kind: Prompt
|
|
name: multi-tool-agent
|
|
description: Agent with multiple tools
|
|
model:
|
|
id: gpt-4
|
|
tools:
|
|
- kind: web_search
|
|
name: search
|
|
description: Search the web
|
|
- kind: function
|
|
name: get_weather
|
|
description: Get weather information
|
|
- kind: code_interpreter
|
|
name: code
|
|
description: Execute code
|
|
"""
|
|
result = agent_schema_dispatch(yaml.safe_load(yaml_content))
|
|
|
|
assert isinstance(result, PromptAgent)
|
|
assert result.name == "multi-tool-agent"
|
|
assert len(result.tools) == 3
|
|
# Tools are polymorphically created based on their kind
|
|
assert result.tools[0].kind == "web_search"
|
|
assert result.tools[1].kind == "function"
|
|
assert result.tools[2].kind == "code_interpreter"
|
|
|
|
|
|
def test_agent_schema_dispatch_model_resource():
|
|
"""Test loading a model resource."""
|
|
yaml_content = """
|
|
kind: Model
|
|
name: my-model
|
|
id: gpt-4
|
|
"""
|
|
result = agent_schema_dispatch(yaml.safe_load(yaml_content))
|
|
|
|
assert isinstance(result, ModelResource)
|
|
assert result.id == "gpt-4"
|
|
|
|
|
|
def test_agent_schema_dispatch_property_schema_with_nested_properties():
|
|
"""Test loading a property schema with nested properties."""
|
|
yaml_content = """
|
|
kind: property_schema
|
|
strict: true
|
|
properties:
|
|
- kind: property
|
|
name: name
|
|
description: User name
|
|
- kind: object
|
|
name: address
|
|
description: User address
|
|
properties:
|
|
- kind: property
|
|
name: street
|
|
description: Street address
|
|
- kind: property
|
|
name: city
|
|
description: City name
|
|
- kind: array
|
|
name: tags
|
|
description: User tags
|
|
"""
|
|
result = agent_schema_dispatch(yaml.safe_load(yaml_content))
|
|
|
|
assert isinstance(result, PropertySchema)
|
|
assert result.strict is True
|
|
assert len(result.properties) == 3
|
|
# Properties are polymorphically created based on their kind
|
|
assert result.properties[0].kind == "property"
|
|
assert result.properties[1].kind == "object"
|
|
assert result.properties[2].kind == "array"
|
|
|
|
|
|
def _get_agent_sample_yaml_files() -> list[tuple[Path, Path]]:
|
|
"""Helper function to collect all YAML files from agent-samples directory."""
|
|
current_file = Path(__file__)
|
|
repo_root = current_file.parent.parent.parent.parent # tests -> declarative -> packages -> python
|
|
agent_samples_dir = repo_root.parent / "agent-samples"
|
|
|
|
if not agent_samples_dir.exists():
|
|
return []
|
|
|
|
yaml_files = list(agent_samples_dir.rglob("*.yaml")) + list(agent_samples_dir.rglob("*.yml"))
|
|
return [(yaml_file, agent_samples_dir) for yaml_file in yaml_files]
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"yaml_file,agent_samples_dir",
|
|
_get_agent_sample_yaml_files(),
|
|
ids=lambda x: x[0].name if isinstance(x, tuple) else str(x),
|
|
)
|
|
def test_agent_schema_dispatch_agent_samples(yaml_file: Path, agent_samples_dir: Path):
|
|
"""Test that agent_schema_dispatch successfully loads a YAML file from agent-samples directory."""
|
|
with open(yaml_file) as f:
|
|
content = f.read()
|
|
result = agent_schema_dispatch(yaml.safe_load(content))
|
|
# Result can be None for unknown kinds, but should not raise exceptions
|
|
assert result is not None, f"agent_schema_dispatch returned None for {yaml_file.relative_to(agent_samples_dir)}"
|
|
|
|
|
|
class TestAgentFactoryCreateFromDict:
|
|
"""Tests for AgentFactory.create_agent_from_dict method."""
|
|
|
|
def test_create_agent_from_dict_parses_prompt_agent(self):
|
|
"""Test that create_agent_from_dict correctly parses a PromptAgent definition."""
|
|
from unittest.mock import MagicMock
|
|
|
|
from agent_framework_declarative import AgentFactory
|
|
|
|
agent_def = {
|
|
"kind": "Prompt",
|
|
"name": "TestAgent",
|
|
"description": "A test agent",
|
|
"instructions": "You are a helpful assistant.",
|
|
}
|
|
|
|
# Use a pre-configured chat client to avoid needing model
|
|
mock_client = MagicMock()
|
|
mock_client.create_agent.return_value = MagicMock()
|
|
|
|
factory = AgentFactory(chat_client=mock_client)
|
|
agent = factory.create_agent_from_dict(agent_def)
|
|
|
|
assert agent is not None
|
|
|
|
def test_create_agent_from_dict_matches_yaml(self):
|
|
"""Test that create_agent_from_dict produces same result as create_agent_from_yaml."""
|
|
from unittest.mock import MagicMock
|
|
|
|
from agent_framework_declarative import AgentFactory
|
|
|
|
yaml_content = """
|
|
kind: Prompt
|
|
name: TestAgent
|
|
description: A test agent
|
|
instructions: You are a helpful assistant.
|
|
"""
|
|
|
|
agent_def = {
|
|
"kind": "Prompt",
|
|
"name": "TestAgent",
|
|
"description": "A test agent",
|
|
"instructions": "You are a helpful assistant.",
|
|
}
|
|
|
|
# Use a pre-configured chat client to avoid needing model
|
|
mock_client = MagicMock()
|
|
mock_client.create_agent.return_value = MagicMock()
|
|
|
|
factory = AgentFactory(chat_client=mock_client)
|
|
|
|
# Create from YAML string
|
|
agent_from_yaml = factory.create_agent_from_yaml(yaml_content)
|
|
|
|
# Create from dict
|
|
agent_from_dict = factory.create_agent_from_dict(agent_def)
|
|
|
|
# Both should produce agents with same name
|
|
assert agent_from_yaml.name == agent_from_dict.name
|
|
assert agent_from_yaml.description == agent_from_dict.description
|
|
|
|
def test_create_agent_from_dict_invalid_kind_raises(self):
|
|
"""Test that non-PromptAgent kind raises DeclarativeLoaderError."""
|
|
from agent_framework_declarative import AgentFactory
|
|
from agent_framework_declarative._loader import DeclarativeLoaderError
|
|
|
|
# Resource kind (not PromptAgent)
|
|
agent_def = {
|
|
"kind": "Resource",
|
|
"name": "TestResource",
|
|
}
|
|
|
|
factory = AgentFactory()
|
|
with pytest.raises(DeclarativeLoaderError, match="Only definitions for a PromptAgent are supported"):
|
|
factory.create_agent_from_dict(agent_def)
|
|
|
|
def test_create_agent_from_dict_without_model_or_client_raises(self):
|
|
"""Test that missing both model and chat_client raises DeclarativeLoaderError."""
|
|
from agent_framework_declarative import AgentFactory
|
|
from agent_framework_declarative._loader import DeclarativeLoaderError
|
|
|
|
agent_def = {
|
|
"kind": "Prompt",
|
|
"name": "TestAgent",
|
|
"instructions": "You are helpful.",
|
|
}
|
|
|
|
factory = AgentFactory()
|
|
with pytest.raises(DeclarativeLoaderError, match="ChatClient must be provided"):
|
|
factory.create_agent_from_dict(agent_def)
|
|
|
|
|
|
class TestAgentFactorySafeMode:
|
|
"""Tests for AgentFactory safe_mode parameter."""
|
|
|
|
def test_agent_factory_safe_mode_default_is_true(self):
|
|
"""Test that safe_mode is True by default."""
|
|
from agent_framework_declarative._loader import AgentFactory
|
|
|
|
factory = AgentFactory()
|
|
assert factory.safe_mode is True
|
|
|
|
def test_agent_factory_safe_mode_can_be_set_false(self):
|
|
"""Test that safe_mode can be explicitly set to False."""
|
|
from agent_framework_declarative._loader import AgentFactory
|
|
|
|
factory = AgentFactory(safe_mode=False)
|
|
assert factory.safe_mode is False
|
|
|
|
def test_agent_factory_safe_mode_blocks_env_in_yaml(self, monkeypatch):
|
|
"""Test that safe_mode=True blocks environment variable access in YAML parsing."""
|
|
from unittest.mock import MagicMock
|
|
|
|
from agent_framework_declarative._loader import AgentFactory
|
|
|
|
monkeypatch.setenv("TEST_MODEL_ID", "gpt-4-from-env")
|
|
|
|
# Create a mock chat client to avoid needing real provider
|
|
mock_client = MagicMock()
|
|
|
|
yaml_content = """
|
|
kind: Prompt
|
|
name: test-agent
|
|
description: =Env.TEST_DESCRIPTION
|
|
instructions: Hello world
|
|
"""
|
|
monkeypatch.setenv("TEST_DESCRIPTION", "Description from env")
|
|
|
|
# With safe_mode=True (default), Env access should fail and return original value
|
|
factory = AgentFactory(chat_client=mock_client, safe_mode=True)
|
|
agent = factory.create_agent_from_yaml(yaml_content)
|
|
|
|
# The description should NOT be resolved from env (PowerFx fails, returns original)
|
|
assert agent.description == "=Env.TEST_DESCRIPTION"
|
|
|
|
@pytest.mark.skipif(not _powerfx_available, reason="PowerFx engine not available")
|
|
def test_agent_factory_safe_mode_false_allows_env_in_yaml(self, monkeypatch):
|
|
"""Test that safe_mode=False allows environment variable access in YAML parsing."""
|
|
from unittest.mock import MagicMock
|
|
|
|
from agent_framework_declarative._loader import AgentFactory
|
|
|
|
monkeypatch.setenv("TEST_DESCRIPTION", "Description from env")
|
|
|
|
# Create a mock chat client to avoid needing real provider
|
|
mock_client = MagicMock()
|
|
|
|
yaml_content = """
|
|
kind: Prompt
|
|
name: test-agent
|
|
description: =Env.TEST_DESCRIPTION
|
|
instructions: Hello world
|
|
"""
|
|
|
|
# With safe_mode=False, Env access should work
|
|
factory = AgentFactory(chat_client=mock_client, safe_mode=False)
|
|
agent = factory.create_agent_from_yaml(yaml_content)
|
|
|
|
# The description should be resolved from env
|
|
assert agent.description == "Description from env"
|
|
|
|
def test_agent_factory_safe_mode_with_api_key_connection(self, monkeypatch):
|
|
"""Test safe_mode with API key connection containing env variable."""
|
|
from agent_framework_declarative._models import _safe_mode_context
|
|
|
|
monkeypatch.setenv("MY_API_KEY", "secret-key-123")
|
|
|
|
yaml_content = """
|
|
kind: Prompt
|
|
name: test-agent
|
|
description: Test agent
|
|
instructions: Hello
|
|
model:
|
|
id: gpt-4
|
|
provider: OpenAI
|
|
apiType: Chat
|
|
connection:
|
|
kind: key
|
|
apiKey: =Env.MY_API_KEY
|
|
"""
|
|
|
|
# Manually trigger the YAML parsing to check the context is set correctly
|
|
import yaml as yaml_module
|
|
|
|
from agent_framework_declarative._models import agent_schema_dispatch
|
|
|
|
token = _safe_mode_context.set(True) # Ensure we're in safe mode
|
|
try:
|
|
result = agent_schema_dispatch(yaml_module.safe_load(yaml_content))
|
|
|
|
# The API key should NOT be resolved (still has the PowerFx expression)
|
|
assert result.model.connection.apiKey == "=Env.MY_API_KEY"
|
|
finally:
|
|
_safe_mode_context.reset(token)
|
|
|
|
@pytest.mark.skipif(not _powerfx_available, reason="PowerFx engine not available")
|
|
def test_agent_factory_safe_mode_false_resolves_api_key(self, monkeypatch):
|
|
"""Test safe_mode=False resolves API key from environment."""
|
|
from agent_framework_declarative._models import _safe_mode_context
|
|
|
|
monkeypatch.setenv("MY_API_KEY", "secret-key-123")
|
|
|
|
yaml_content = """
|
|
kind: Prompt
|
|
name: test-agent
|
|
description: Test agent
|
|
instructions: Hello
|
|
model:
|
|
id: gpt-4
|
|
provider: OpenAI
|
|
apiType: Chat
|
|
connection:
|
|
kind: key
|
|
apiKey: =Env.MY_API_KEY
|
|
"""
|
|
|
|
# With safe_mode=False, the API key should be resolved
|
|
import yaml as yaml_module
|
|
|
|
from agent_framework_declarative._models import agent_schema_dispatch
|
|
|
|
token = _safe_mode_context.set(False) # Disable safe mode
|
|
try:
|
|
result = agent_schema_dispatch(yaml_module.safe_load(yaml_content))
|
|
|
|
# The API key should be resolved from environment
|
|
assert result.model.connection.apiKey == "secret-key-123"
|
|
finally:
|
|
_safe_mode_context.reset(token)
|