mirror of
https://github.com/microsoft/agent-framework.git
synced 2026-06-16 21:04:09 +08:00
b05fc9e849
* Initial plan * Add load_dotenv() to 303 Python samples for environment variable loading Co-authored-by: TaoChenOSU <12570346+TaoChenOSU@users.noreply.github.com> * Update SAMPLE_GUIDELINES.md to document load_dotenv() requirement Co-authored-by: TaoChenOSU <12570346+TaoChenOSU@users.noreply.github.com> * Update samples README.md to document .env file usage Co-authored-by: TaoChenOSU <12570346+TaoChenOSU@users.noreply.github.com> * Run ruff format on all changed sample files Co-authored-by: TaoChenOSU <12570346+TaoChenOSU@users.noreply.github.com> * Clarify load_dotenv() usage in README - local dev vs production Co-authored-by: TaoChenOSU <12570346+TaoChenOSU@users.noreply.github.com> * Remove deprecated getting_started folder as requested Co-authored-by: TaoChenOSU <12570346+TaoChenOSU@users.noreply.github.com> * Document env_file_path parameter for per-client configuration Co-authored-by: TaoChenOSU <12570346+TaoChenOSU@users.noreply.github.com> * Merge main branch to resolve conflicts Co-authored-by: TaoChenOSU <12570346+TaoChenOSU@users.noreply.github.com> * Fix run_evaluation.py file that was empty in merge commit Co-authored-by: TaoChenOSU <12570346+TaoChenOSU@users.noreply.github.com> * Remove dotnet changes from merge - out of scope for this PR Co-authored-by: TaoChenOSU <12570346+TaoChenOSU@users.noreply.github.com> * Remove package and test changes from merge - only sample changes needed Co-authored-by: TaoChenOSU <12570346+TaoChenOSU@users.noreply.github.com> * Remove test_func_utils.py - only sample changes needed Co-authored-by: TaoChenOSU <12570346+TaoChenOSU@users.noreply.github.com> * Revert sample files not in original changeset - keep only load_dotenv additions Co-authored-by: TaoChenOSU <12570346+TaoChenOSU@users.noreply.github.com> * Move load_dotenv() outside snippet tag in 06_host_your_agent.py Co-authored-by: TaoChenOSU <12570346+TaoChenOSU@users.noreply.github.com> * Fix comment placement - move load_dotenv before code comments Co-authored-by: TaoChenOSU <12570346+TaoChenOSU@users.noreply.github.com> * Fix load_dotenv() placement across all samples - after docstring, before code comments Co-authored-by: TaoChenOSU <12570346+TaoChenOSU@users.noreply.github.com> * Merge latest main branch with load_dotenv changes Co-authored-by: TaoChenOSU <12570346+TaoChenOSU@users.noreply.github.com> * Remove non-sample changes from merge - keep only load_dotenv additions Co-authored-by: TaoChenOSU <12570346+TaoChenOSU@users.noreply.github.com> * Revert non-load_dotenv sample changes from merge Co-authored-by: TaoChenOSU <12570346+TaoChenOSU@users.noreply.github.com> * Fix run_evaluation.py - use main's improved version (file already had load_dotenv) Co-authored-by: TaoChenOSU <12570346+TaoChenOSU@users.noreply.github.com> * Manual update * Manual update 2 * Fix Role usage and load_dotenv placement per PR review feedback Co-authored-by: eavanvalkenburg <13749212+eavanvalkenburg@users.noreply.github.com> * Fix Role usage - use string literals not enum attributes Co-authored-by: eavanvalkenburg <13749212+eavanvalkenburg@users.noreply.github.com> * Fix SAMPLE_GUIDELINES.md example - load_dotenv before docstring per guidance Co-authored-by: eavanvalkenburg <13749212+eavanvalkenburg@users.noreply.github.com> * Move load_dotenv() before docstrings in all samples per SAMPLE_GUIDELINES ordering Co-authored-by: eavanvalkenburg <13749212+eavanvalkenburg@users.noreply.github.com> * Address PR review: rename files, fix placement, add session usage, remove note Co-authored-by: eavanvalkenburg <13749212+eavanvalkenburg@users.noreply.github.com> * Update Redis README to reference renamed file redis_history_provider.py Co-authored-by: eavanvalkenburg <13749212+eavanvalkenburg@users.noreply.github.com> --------- Co-authored-by: copilot-swe-agent[bot] <198982749+Copilot@users.noreply.github.com> Co-authored-by: TaoChenOSU <12570346+TaoChenOSU@users.noreply.github.com> Co-authored-by: Tao Chen <taochen@microsoft.com> Co-authored-by: eavanvalkenburg <13749212+eavanvalkenburg@users.noreply.github.com> Co-authored-by: Eduard van Valkenburg <eavanvalkenburg@users.noreply.github.com>
328 lines
11 KiB
Python
328 lines
11 KiB
Python
# Copyright (c) Microsoft. All rights reserved.
|
|
"""Purview policy enforcement sample (Python).
|
|
|
|
Shows:
|
|
1. Creating a basic chat agent
|
|
2. Adding Purview policy evaluation via AGENT middleware (agent-level)
|
|
3. Adding Purview policy evaluation via CHAT middleware (chat-client level)
|
|
4. Implementing a custom cache provider for advanced caching scenarios
|
|
5. Running threaded conversations and printing results
|
|
|
|
Note: Caching is automatic and enabled by default.
|
|
|
|
Environment variables:
|
|
- AZURE_OPENAI_ENDPOINT (required)
|
|
- AZURE_OPENAI_DEPLOYMENT_NAME (optional, defaults to gpt-4o-mini)
|
|
- PURVIEW_CLIENT_APP_ID (required)
|
|
- PURVIEW_USE_CERT_AUTH (optional, set to "true" for certificate auth)
|
|
- PURVIEW_TENANT_ID (required if certificate auth)
|
|
- PURVIEW_CERT_PATH (required if certificate auth)
|
|
- PURVIEW_CERT_PASSWORD (optional)
|
|
- PURVIEW_DEFAULT_USER_ID (optional, user ID for Purview evaluation)
|
|
"""
|
|
|
|
import asyncio
|
|
import os
|
|
from typing import Any
|
|
|
|
from agent_framework import Agent, AgentResponse, Message
|
|
from agent_framework.azure import AzureOpenAIChatClient
|
|
from agent_framework.microsoft import (
|
|
PurviewChatPolicyMiddleware,
|
|
PurviewPolicyMiddleware,
|
|
PurviewSettings,
|
|
)
|
|
from azure.identity import (
|
|
AzureCliCredential,
|
|
CertificateCredential,
|
|
InteractiveBrowserCredential,
|
|
)
|
|
from dotenv import load_dotenv
|
|
|
|
# Load environment variables from .env file
|
|
load_dotenv()
|
|
|
|
JOKER_NAME = "Joker"
|
|
JOKER_INSTRUCTIONS = "You are good at telling jokes. Keep responses concise."
|
|
|
|
|
|
# Custom Cache Provider Implementation
|
|
class SimpleDictCacheProvider:
|
|
"""A simple custom cache provider that stores everything in a dictionary.
|
|
|
|
This example demonstrates how to implement the CacheProvider protocol.
|
|
"""
|
|
|
|
def __init__(self) -> None:
|
|
"""Initialize the simple dictionary cache."""
|
|
self._cache: dict[str, Any] = {}
|
|
self._access_count: dict[str, int] = {}
|
|
|
|
async def get(self, key: str) -> Any | None:
|
|
"""Get a value from the cache.
|
|
|
|
Args:
|
|
key: The cache key.
|
|
|
|
Returns:
|
|
The cached value or None if not found.
|
|
"""
|
|
value = self._cache.get(key)
|
|
if value is not None:
|
|
self._access_count[key] = self._access_count.get(key, 0) + 1
|
|
print(f"[CustomCache] Cache HIT for key: {key[:50]}... (accessed {self._access_count[key]} times)")
|
|
else:
|
|
print(f"[CustomCache] Cache MISS for key: {key[:50]}...")
|
|
return value
|
|
|
|
async def set(self, key: str, value: Any, ttl_seconds: int | None = None) -> None:
|
|
"""Set a value in the cache.
|
|
|
|
Args:
|
|
key: The cache key.
|
|
value: The value to cache.
|
|
ttl_seconds: Time to live in seconds (ignored in this simple implementation).
|
|
"""
|
|
self._cache[key] = value
|
|
print(f"[CustomCache] Cached value for key: {key[:50]}... (TTL: {ttl_seconds}s)")
|
|
|
|
async def remove(self, key: str) -> None:
|
|
"""Remove a value from the cache.
|
|
|
|
Args:
|
|
key: The cache key.
|
|
"""
|
|
if key in self._cache:
|
|
del self._cache[key]
|
|
self._access_count.pop(key, None)
|
|
print(f"[CustomCache] Removed key: {key[:50]}...")
|
|
|
|
|
|
def _get_env(name: str, *, required: bool = True, default: str | None = None) -> str:
|
|
val = os.environ.get(name, default)
|
|
if required and not val:
|
|
raise RuntimeError(f"Environment variable {name} is required")
|
|
return val # type: ignore[return-value]
|
|
|
|
|
|
def build_credential() -> Any:
|
|
"""Select an Azure credential for Purview authentication.
|
|
|
|
Supported modes:
|
|
1. CertificateCredential (if PURVIEW_USE_CERT_AUTH=true)
|
|
2. InteractiveBrowserCredential (requires PURVIEW_CLIENT_APP_ID)
|
|
"""
|
|
client_id = _get_env("PURVIEW_CLIENT_APP_ID", required=True)
|
|
use_cert_auth = _get_env("PURVIEW_USE_CERT_AUTH", required=False, default="false").lower() == "true"
|
|
|
|
if not client_id:
|
|
raise RuntimeError(
|
|
"PURVIEW_CLIENT_APP_ID is required for interactive browser authentication; "
|
|
"set PURVIEW_USE_CERT_AUTH=true for certificate mode instead."
|
|
)
|
|
|
|
if use_cert_auth:
|
|
tenant_id = _get_env("PURVIEW_TENANT_ID")
|
|
cert_path = _get_env("PURVIEW_CERT_PATH")
|
|
cert_password = _get_env("PURVIEW_CERT_PASSWORD", required=False, default=None)
|
|
print(f"Using Certificate Authentication (tenant: {tenant_id}, cert: {cert_path})")
|
|
return CertificateCredential(
|
|
tenant_id=tenant_id,
|
|
client_id=client_id,
|
|
certificate_path=cert_path,
|
|
password=cert_password,
|
|
)
|
|
|
|
print(f"Using Interactive Browser Authentication (client_id: {client_id})")
|
|
return InteractiveBrowserCredential(client_id=client_id)
|
|
|
|
|
|
async def run_with_agent_middleware() -> None:
|
|
endpoint = os.environ.get("AZURE_OPENAI_ENDPOINT")
|
|
if not endpoint:
|
|
print("Skipping run: AZURE_OPENAI_ENDPOINT not set")
|
|
return
|
|
|
|
deployment = os.environ.get("AZURE_OPENAI_DEPLOYMENT_NAME", "gpt-4o-mini")
|
|
user_id = os.environ.get("PURVIEW_DEFAULT_USER_ID")
|
|
client = AzureOpenAIChatClient(deployment_name=deployment, endpoint=endpoint, credential=AzureCliCredential())
|
|
|
|
purview_agent_middleware = PurviewPolicyMiddleware(
|
|
build_credential(),
|
|
PurviewSettings(
|
|
app_name="Agent Framework Sample App",
|
|
),
|
|
)
|
|
|
|
agent = Agent(
|
|
client=client,
|
|
instructions=JOKER_INSTRUCTIONS,
|
|
name=JOKER_NAME,
|
|
middleware=[purview_agent_middleware],
|
|
)
|
|
|
|
print("-- Agent MiddlewareTypes Path --")
|
|
first: AgentResponse = await agent.run(
|
|
Message("user", ["Tell me a joke about a pirate."], additional_properties={"user_id": user_id})
|
|
)
|
|
print("First response (agent middleware):\n", first)
|
|
|
|
second: AgentResponse = await agent.run(
|
|
Message(role="user", text="That was funny. Tell me another one.", additional_properties={"user_id": user_id})
|
|
)
|
|
print("Second response (agent middleware):\n", second)
|
|
|
|
|
|
async def run_with_chat_middleware() -> None:
|
|
endpoint = os.environ.get("AZURE_OPENAI_ENDPOINT")
|
|
if not endpoint:
|
|
print("Skipping chat middleware run: AZURE_OPENAI_ENDPOINT not set")
|
|
return
|
|
|
|
deployment = os.environ.get("AZURE_OPENAI_DEPLOYMENT_NAME", default="gpt-4o-mini")
|
|
user_id = os.environ.get("PURVIEW_DEFAULT_USER_ID")
|
|
|
|
client = AzureOpenAIChatClient(
|
|
deployment_name=deployment,
|
|
endpoint=endpoint,
|
|
credential=AzureCliCredential(),
|
|
middleware=[
|
|
PurviewChatPolicyMiddleware(
|
|
build_credential(),
|
|
PurviewSettings(
|
|
app_name="Agent Framework Sample App (Chat)",
|
|
),
|
|
)
|
|
],
|
|
)
|
|
|
|
agent = Agent(
|
|
client=client,
|
|
instructions=JOKER_INSTRUCTIONS,
|
|
name=JOKER_NAME,
|
|
)
|
|
|
|
print("-- Chat MiddlewareTypes Path --")
|
|
first: AgentResponse = await agent.run(
|
|
Message(
|
|
role="user",
|
|
text="Give me a short clean joke.",
|
|
additional_properties={"user_id": user_id},
|
|
)
|
|
)
|
|
print("First response (chat middleware):\n", first)
|
|
|
|
second: AgentResponse = await agent.run(
|
|
Message(
|
|
role="user",
|
|
text="One more please.",
|
|
additional_properties={"user_id": user_id},
|
|
)
|
|
)
|
|
print("Second response (chat middleware):\n", second)
|
|
|
|
|
|
async def run_with_custom_cache_provider() -> None:
|
|
"""Demonstrate implementing and using a custom cache provider."""
|
|
endpoint = os.environ.get("AZURE_OPENAI_ENDPOINT")
|
|
if not endpoint:
|
|
print("Skipping custom cache provider run: AZURE_OPENAI_ENDPOINT not set")
|
|
return
|
|
|
|
deployment = os.environ.get("AZURE_OPENAI_DEPLOYMENT_NAME", "gpt-4o-mini")
|
|
user_id = os.environ.get("PURVIEW_DEFAULT_USER_ID")
|
|
client = AzureOpenAIChatClient(deployment_name=deployment, endpoint=endpoint, credential=AzureCliCredential())
|
|
|
|
custom_cache = SimpleDictCacheProvider()
|
|
|
|
purview_agent_middleware = PurviewPolicyMiddleware(
|
|
build_credential(),
|
|
PurviewSettings(
|
|
app_name="Agent Framework Sample App (Custom Provider)",
|
|
),
|
|
cache_provider=custom_cache,
|
|
)
|
|
|
|
agent = Agent(
|
|
client=client,
|
|
instructions=JOKER_INSTRUCTIONS,
|
|
name=JOKER_NAME,
|
|
middleware=[purview_agent_middleware],
|
|
)
|
|
|
|
print("-- Custom Cache Provider Path --")
|
|
print("Using SimpleDictCacheProvider")
|
|
|
|
first: AgentResponse = await agent.run(
|
|
Message(role="user", text="Tell me a joke about a programmer.", additional_properties={"user_id": user_id})
|
|
)
|
|
print("First response (custom provider):\n", first)
|
|
|
|
second: AgentResponse = await agent.run(
|
|
Message("user", ["That's hilarious! One more?"], additional_properties={"user_id": user_id})
|
|
)
|
|
print("Second response (custom provider):\n", second)
|
|
|
|
"""Demonstrate using the default built-in cache."""
|
|
endpoint = os.environ.get("AZURE_OPENAI_ENDPOINT")
|
|
if not endpoint:
|
|
print("Skipping default cache run: AZURE_OPENAI_ENDPOINT not set")
|
|
return
|
|
|
|
deployment = os.environ.get("AZURE_OPENAI_DEPLOYMENT_NAME", "gpt-4o-mini")
|
|
user_id = os.environ.get("PURVIEW_DEFAULT_USER_ID")
|
|
client = AzureOpenAIChatClient(deployment_name=deployment, endpoint=endpoint, credential=AzureCliCredential())
|
|
|
|
# No cache_provider specified - uses default InMemoryCacheProvider
|
|
purview_agent_middleware = PurviewPolicyMiddleware(
|
|
build_credential(),
|
|
PurviewSettings(
|
|
app_name="Agent Framework Sample App (Default Cache)",
|
|
cache_ttl_seconds=3600,
|
|
max_cache_size_bytes=100 * 1024 * 1024, # 100MB
|
|
),
|
|
)
|
|
|
|
agent = Agent(
|
|
client=client,
|
|
instructions=JOKER_INSTRUCTIONS,
|
|
name=JOKER_NAME,
|
|
middleware=[purview_agent_middleware],
|
|
)
|
|
|
|
print("-- Default Cache Path --")
|
|
print("Using default InMemoryCacheProvider with settings-based configuration")
|
|
|
|
first: AgentResponse = await agent.run(
|
|
Message("user", ["Tell me a joke about AI."], additional_properties={"user_id": user_id})
|
|
)
|
|
print("First response (default cache):\n", first)
|
|
|
|
second: AgentResponse = await agent.run(
|
|
Message("user", ["Nice! Another AI joke please."], additional_properties={"user_id": user_id})
|
|
)
|
|
print("Second response (default cache):\n", second)
|
|
|
|
|
|
async def main() -> None:
|
|
print("== Purview Agent Sample (MiddlewareTypes with Automatic Caching) ==")
|
|
|
|
try:
|
|
await run_with_agent_middleware()
|
|
except Exception as ex: # pragma: no cover - demo resilience
|
|
print(f"Agent middleware path failed: {ex}")
|
|
|
|
try:
|
|
await run_with_chat_middleware()
|
|
except Exception as ex: # pragma: no cover - demo resilience
|
|
print(f"Chat middleware path failed: {ex}")
|
|
|
|
try:
|
|
await run_with_custom_cache_provider()
|
|
except Exception as ex: # pragma: no cover - demo resilience
|
|
print(f"Custom cache provider path failed: {ex}")
|
|
|
|
|
|
if __name__ == "__main__":
|
|
asyncio.run(main())
|