diff --git a/.devcontainer/dotnet/devcontainer.json b/.devcontainer/dotnet/devcontainer.json index a10557f981..59b56a4438 100644 --- a/.devcontainer/dotnet/devcontainer.json +++ b/.devcontainer/dotnet/devcontainer.json @@ -1,10 +1,11 @@ { "name": "C# (.NET)", - "image": "mcr.microsoft.com/devcontainers/dotnet:9.0", + "image": "mcr.microsoft.com/devcontainers/dotnet:10.0", "features": { "ghcr.io/devcontainers/features/dotnet:2.4.0": {}, "ghcr.io/devcontainers/features/powershell:1.5.1": {}, - "ghcr.io/devcontainers/features/azure-cli:1.2.8": {} + "ghcr.io/devcontainers/features/azure-cli:1.2.8": {}, + "ghcr.io/devcontainers/features/docker-in-docker:2.12.4": {} }, "workspaceFolder": "/workspaces/agent-framework/dotnet/", "customizations": { diff --git a/.github/CODEOWNERS b/.github/CODEOWNERS new file mode 100644 index 0000000000..66023c649a --- /dev/null +++ b/.github/CODEOWNERS @@ -0,0 +1,4 @@ +# Code ownership assignments +# https://docs.github.com/repositories/managing-your-repositorys-settings-and-features/customizing-your-repository/about-code-owners + +python/packages/azurefunctions/ @microsoft/agentframework-durabletask-developers diff --git a/.github/actions/azure-functions-integration-setup/action.yml b/.github/actions/azure-functions-integration-setup/action.yml new file mode 100644 index 0000000000..6be5afb814 --- /dev/null +++ b/.github/actions/azure-functions-integration-setup/action.yml @@ -0,0 +1,36 @@ +name: Azure Functions Integration Test Setup +description: Prepare local emulators and tools for Azure Functions integration tests + +runs: + using: "composite" + steps: + - name: Start Durable Task Scheduler Emulator + shell: bash + run: | + if [ "$(docker ps -aq -f name=dts-emulator)" ]; then + echo "Stopping and removing existing Durable Task Scheduler Emulator" + docker rm -f dts-emulator + fi + echo "Starting Durable Task Scheduler Emulator" + docker run -d --name dts-emulator -p 8080:8080 -p 8082:8082 mcr.microsoft.com/dts/dts-emulator:latest + echo "Waiting for Durable Task Scheduler Emulator to be ready" + timeout 30 bash -c 'until curl --silent http://localhost:8080/healthz; do sleep 1; done' + echo "Durable Task Scheduler Emulator is ready" + - name: Start Azurite (Azure Storage emulator) + shell: bash + run: | + if [ "$(docker ps -aq -f name=azurite)" ]; then + echo "Stopping and removing existing Azurite (Azure Storage emulator)" + docker rm -f azurite + fi + echo "Starting Azurite (Azure Storage emulator)" + docker run -d --name azurite -p 10000:10000 -p 10001:10001 -p 10002:10002 mcr.microsoft.com/azure-storage/azurite + echo "Waiting for Azurite (Azure Storage emulator) to be ready" + timeout 30 bash -c 'until curl --silent http://localhost:10000/devstoreaccount1; do sleep 1; done' + echo "Azurite (Azure Storage emulator) is ready" + - name: Install Azure Functions Core Tools + shell: bash + run: | + echo "Installing Azure Functions Core Tools" + npm install -g azure-functions-core-tools@4 --unsafe-perm true + func --version diff --git a/.github/dependabot.yml b/.github/dependabot.yml index 6ea60a0d59..90b127a829 100644 --- a/.github/dependabot.yml +++ b/.github/dependabot.yml @@ -11,9 +11,6 @@ updates: schedule: interval: "cron" cronjob: "0 8 * * 4,0" # Every Thursday(4) and Sunday(0) at 8:00 UTC - experimental: - nuget-native-updater: false - enable-cooldown-metrics-collection: false ignore: # For all System.* and Microsoft.Extensions/Bcl.* packages, ignore all major version updates - dependency-name: "System.*" @@ -28,6 +25,14 @@ updates: - "dependencies" # Maintain dependencies for python + - package-ecosystem: "pip" + directory: "python/" + schedule: + interval: "weekly" + day: "monday" + labels: + - "python" + - "dependencies" - package-ecosystem: "uv" directory: "python/" schedule: diff --git a/.github/instructions/durabletask-dotnet.instructions.md b/.github/instructions/durabletask-dotnet.instructions.md new file mode 100644 index 0000000000..84aeb542ea --- /dev/null +++ b/.github/instructions/durabletask-dotnet.instructions.md @@ -0,0 +1,17 @@ +--- +applyTo: "dotnet/src/Microsoft.Agents.AI.DurableTask/**,dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/**" +--- + +# Durable Task area code instructions + +The following guidelines apply to pull requests that modify files under +`dotnet/src/Microsoft.Agents.AI.DurableTask/**` or +`dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/**`: + +## CHANGELOG.md + +- Each pull request that modifies code should add just one bulleted entry to the `CHANGELOG.md` file containing a change title (usually the PR title) and a link to the PR itself. +- New PRs should be added to the top of the `CHANGELOG.md` file under a "## [Unreleased]" heading. +- If the PR is the first since the last release, the existing "## [Unreleased]" heading should be replaced with a "## v[X.Y.Z]" heading and the PRs since the last release should be added to the new "## [Unreleased]" heading. +- The style of new `CHANGELOG.md` entries should match the style of the other entries in the file. +- If the PR introduces a breaking change, the changelog entry should be prefixed with "[BREAKING]". diff --git a/.github/upgrades/prompts/SemanticKernelToAgentFramework.md b/.github/upgrades/prompts/SemanticKernelToAgentFramework.md index a121a5f446..a8c3dcb0a6 100644 --- a/.github/upgrades/prompts/SemanticKernelToAgentFramework.md +++ b/.github/upgrades/prompts/SemanticKernelToAgentFramework.md @@ -142,9 +142,9 @@ Replace these Semantic Kernel agent classes with their Agent Framework equivalen |----------------------|----------------------------|-------------------| | `IChatCompletionService` | `IChatClient` | Convert to `IChatClient` using `chatService.AsChatClient()` extensions | | `ChatCompletionAgent` | `ChatClientAgent` | Remove `Kernel` parameter, add `IChatClient` parameter | -| `OpenAIAssistantAgent` | `AIAgent` (via extension) | **New**: `OpenAIClient.GetAssistantClient().CreateAIAgent()`
**Existing**: `OpenAIClient.GetAssistantClient().GetAIAgent(assistantId)` | +| `OpenAIAssistantAgent` | `AIAgent` (via extension) | ⚠️ **Deprecated** - Use Responses API instead.
**New**: `OpenAIClient.GetAssistantClient().CreateAIAgent()`
**Existing**: `OpenAIClient.GetAssistantClient().GetAIAgent(assistantId)` | | `AzureAIAgent` | `AIAgent` (via extension) | **New**: `PersistentAgentsClient.CreateAIAgent()`
**Existing**: `PersistentAgentsClient.GetAIAgent(agentId)` | -| `OpenAIResponseAgent` | `AIAgent` (via extension) | Replace with `OpenAIClient.GetOpenAIResponseClient().CreateAIAgent()` | +| `OpenAIResponseAgent` | `AIAgent` (via extension) | Replace with `OpenAIClient.GetOpenAIResponseClient(modelId).CreateAIAgent()` | | `A2AAgent` | `AIAgent` (via extension) | Replace with `A2ACardResolver.GetAIAgentAsync()` | | `BedrockAgent` | Not supported | Custom implementation required | @@ -529,14 +529,14 @@ AIAgent agent = new OpenAIClient(apiKey) .CreateAIAgent(instructions: instructions); ``` -**OpenAI Assistants (New):** +**OpenAI Assistants (New):** ⚠️ *Deprecated - Use Responses API instead* ```csharp AIAgent agent = new OpenAIClient(apiKey) .GetAssistantClient() .CreateAIAgent(modelId, instructions: instructions); ``` -**OpenAI Assistants (Existing):** +**OpenAI Assistants (Existing):** ⚠️ *Deprecated - Use Responses API instead* ```csharp AIAgent agent = new OpenAIClient(apiKey) .GetAssistantClient() @@ -562,6 +562,20 @@ AIAgent agent = await new PersistentAgentsClient(endpoint, credential) .GetAIAgentAsync(agentId); ``` +**OpenAI Responses:** *(Recommended for OpenAI)* +```csharp +AIAgent agent = new OpenAIClient(apiKey) + .GetOpenAIResponseClient(modelId) + .CreateAIAgent(instructions: instructions); +``` + +**Azure OpenAI Responses:** *(Recommended for Azure OpenAI)* +```csharp +AIAgent agent = new AzureOpenAIClient(endpoint, credential) + .GetOpenAIResponseClient(deploymentName) + .CreateAIAgent(instructions: instructions); +``` + **A2A:** ```csharp A2ACardResolver resolver = new(new Uri(agentHost)); @@ -762,35 +776,57 @@ await foreach (var content in agent.InvokeAsync(userInput, thread)) **With this Agent Framework CodeInterpreter pattern:** ```csharp +using System.Text; +using Microsoft.Agents.AI; +using Microsoft.Extensions.AI; + var result = await agent.RunAsync(userInput, thread); Console.WriteLine(result); -// Extract chat response MEAI type via first level breaking glass -var chatResponse = result.RawRepresentation as ChatResponse; +// Get the CodeInterpreterToolCallContent (code input) +CodeInterpreterToolCallContent? toolCallContent = result.Messages + .SelectMany(m => m.Contents) + .OfType() + .FirstOrDefault(); + +if (toolCallContent?.Inputs is not null) +{ + DataContent? codeInput = toolCallContent.Inputs.OfType().FirstOrDefault(); + if (codeInput?.HasTopLevelMediaType("text") ?? false) + { + Console.WriteLine($"Code Input: {Encoding.UTF8.GetString(codeInput.Data.ToArray())}"); + } +} -// Extract underlying SDK updates via second level breaking glass -var underlyingStreamingUpdates = chatResponse?.RawRepresentation as IEnumerable ?? []; +// Get the CodeInterpreterToolResultContent (code output) +CodeInterpreterToolResultContent? toolResultContent = result.Messages + .SelectMany(m => m.Contents) + .OfType() + .FirstOrDefault(); -StringBuilder generatedCode = new(); -foreach (object? underlyingUpdate in underlyingStreamingUpdates ?? []) +if (toolResultContent?.Outputs is not null) { - if (underlyingUpdate is RunStepDetailsUpdate stepDetailsUpdate && stepDetailsUpdate.CodeInterpreterInput is not null) + TextContent? resultOutput = toolResultContent.Outputs.OfType().FirstOrDefault(); + if (resultOutput is not null) { - generatedCode.Append(stepDetailsUpdate.CodeInterpreterInput); + Console.WriteLine($"Code Tool Result: {resultOutput.Text}"); } } -if (!string.IsNullOrEmpty(generatedCode.ToString())) +// Getting any annotations generated by the tool +foreach (AIAnnotation annotation in result.Messages + .SelectMany(m => m.Contents) + .SelectMany(c => c.Annotations ?? [])) { - Console.WriteLine($"\n# {chatResponse?.Messages[0].Role}:Generated Code:\n{generatedCode}"); + Console.WriteLine($"Annotation: {annotation}"); } ``` **Functional differences:** -1. Code interpreter output is separate from text content, not a metadata property -2. Access code via `RunStepDetailsUpdate.CodeInterpreterInput` instead of metadata -3. Use breaking glass pattern to access underlying SDK objects -4. Process text content and code interpreter output independently +1. Code interpreter content is now available via MEAI abstractions - no breaking glass required +2. Use `CodeInterpreterToolCallContent` to access code inputs (the generated code) +3. Use `CodeInterpreterToolResultContent` to access code outputs (execution results) +4. Annotations are accessible via `AIAnnotation` on content items #### Provider-Specific Options Configuration @@ -980,6 +1016,8 @@ AIAgent agent = new AzureOpenAIClient(new Uri(endpoint), new AzureCliCredential( ### 3. OpenAI Assistants Migration +> ⚠️ **DEPRECATION WARNING**: The OpenAI Assistants API has been deprecated. The Agent Framework extension methods for Assistants are marked as `[Obsolete]`. **Please use the Responses API instead** (see Section 6: OpenAI Responses Migration). + **Remove Semantic Kernel Packages:** ```xml @@ -1291,52 +1329,7 @@ var result = await agent.RunAsync(userInput, thread); ``` -### 8. A2A Migration - - -**Remove Semantic Kernel Packages:** -```xml - -``` - -**Add Agent Framework Packages:** -```xml - -``` - - - -**Replace this Semantic Kernel pattern:** -```csharp -using A2A; -using Microsoft.SemanticKernel; -using Microsoft.SemanticKernel.Agents; -using Microsoft.SemanticKernel.Agents.A2A; - -using var httpClient = CreateHttpClient(); -var client = new A2AClient(agentUrl, httpClient); -var cardResolver = new A2ACardResolver(url, httpClient); -var agentCard = await cardResolver.GetAgentCardAsync(); -Console.WriteLine(JsonSerializer.Serialize(agentCard, s_jsonSerializerOptions)); -var agent = new A2AAgent(client, agentCard); -``` - -**With this Agent Framework pattern:** -```csharp -using System; -using A2A; -using Microsoft.Agents.AI; -using Microsoft.Agents.AI.A2A; - -// Initialize an A2ACardResolver to get an A2A agent card. -A2ACardResolver agentCardResolver = new(new Uri(a2aAgentHost)); - -// Create an instance of the AIAgent for an existing A2A agent specified by the agent card. -AIAgent agent = await agentCardResolver.GetAIAgentAsync(); -``` - - -### 9. Unsupported Providers (Require Custom Implementation) +### 8. Unsupported Providers (Require Custom Implementation) #### BedrockAgent Migration @@ -1507,7 +1500,7 @@ Console.WriteLine(result); ``` -### 10. Function Invocation Filtering +### 9. Function Invocation Filtering **Invocation Context** @@ -1615,25 +1608,4 @@ var filteredAgent = originalAgent .Build(); ``` -### 11. Function Invocation Contexts - -**Invocation Context** - -Semantic Kernel's `IAutoFunctionInvocationFilter` provides a `AutoFunctionInvocationContext` where Agent Framework provides `FunctionInvocationContext` - -The property mapping guide from a `AutoFunctionInvocationContext` to a `FunctionInvocationContext` is as follows: -| Semantic Kernel | Agent Framework | -| --- | --- | -| RequestSequenceIndex | Iteration | -| FunctionSequenceIndex | FunctionCallIndex | -| ToolCallId | CallContent.CallId | -| ChatMessageContent | Messages[0] | -| ExecutionSettings | Options | -| ChatHistory | Messages | -| Function | Function | -| Kernel | N/A | -| Result | Use `return` from the delegate | -| Terminate | Terminate | -| CancellationToken | provided via argument to middleware delegate | -| Arguments | Arguments | \ No newline at end of file diff --git a/.github/workflows/codeql-analysis.yml b/.github/workflows/codeql-analysis.yml index 30be1b0e8f..21d3aa2ed0 100644 --- a/.github/workflows/codeql-analysis.yml +++ b/.github/workflows/codeql-analysis.yml @@ -32,7 +32,7 @@ jobs: steps: - name: Checkout repository - uses: actions/checkout@v5 + uses: actions/checkout@v6 with: persist-credentials: false diff --git a/.github/workflows/dotnet-build-and-test.yml b/.github/workflows/dotnet-build-and-test.yml index 5abfe2a879..4a41b343aa 100644 --- a/.github/workflows/dotnet-build-and-test.yml +++ b/.github/workflows/dotnet-build-and-test.yml @@ -8,16 +8,17 @@ name: dotnet-build-and-test on: workflow_dispatch: pull_request: - branches: ["main"] + branches: ["main", "feature*"] merge_group: - branches: ["main"] + branches: ["main", "feature*"] push: - branches: ["main"] + branches: ["main", "feature*"] schedule: - cron: "0 0 * * *" # Run at midnight UTC daily env: COVERAGE_THRESHOLD: 80 + COVERAGE_FRAMEWORK: net10.0 # framework target for which we run/report code coverage concurrency: group: ${{ github.workflow }}-${{ github.event.pull_request.number || github.ref }} @@ -36,7 +37,7 @@ jobs: outputs: dotnetChanges: ${{ steps.filter.outputs.dotnet}} steps: - - uses: actions/checkout@v5 + - uses: actions/checkout@v6 - uses: dorny/paths-filter@v3 id: filter with: @@ -59,26 +60,26 @@ jobs: fail-fast: false matrix: include: - - { targetFramework: "net9.0", os: "ubuntu-latest", configuration: Release, integration-tests: true, environment: "integration" } - - { targetFramework: "net9.0", os: "ubuntu-latest", configuration: Debug } - - { targetFramework: "net9.0", os: "windows-latest", configuration: Release } + - { targetFramework: "net10.0", os: "ubuntu-latest", configuration: Release, integration-tests: true, environment: "integration" } + - { targetFramework: "net9.0", os: "windows-latest", configuration: Debug } + - { targetFramework: "net8.0", os: "ubuntu-latest", configuration: Release } - { targetFramework: "net472", os: "windows-latest", configuration: Release, integration-tests: true, environment: "integration" } runs-on: ${{ matrix.os }} environment: ${{ matrix.environment }} steps: - - uses: actions/checkout@v5 + - uses: actions/checkout@v6 with: - persist-credentials: false - sparse-checkout: | - . - .github - dotnet - python - workflow-samples + persist-credentials: false + sparse-checkout: | + . + .github + dotnet + python + workflow-samples - name: Setup dotnet - uses: actions/setup-dotnet@v5.0.0 + uses: actions/setup-dotnet@v5.0.1 with: global-json-file: ${{ github.workspace }}/dotnet/global.json - name: Build dotnet solutions @@ -123,7 +124,17 @@ jobs: popd rm -rf "$TEMP_DIR" - - name: Run Unit Tests Windows + # Start Cosmos DB Emulator for Cosmos-based unit tests (only on Windows) + - name: Start Azure Cosmos DB Emulator + if: runner.os == 'Windows' + shell: pwsh + run: | + Write-Host "Launching Azure Cosmos DB Emulator" + Import-Module "$env:ProgramFiles\Azure Cosmos DB Emulator\PSModules\Microsoft.Azure.CosmosDB.Emulator" + Start-CosmosDbEmulator -NoUI -Key "C2y6yDjf5/R+ob0N8A7Cgv30VRDJIWEHLM+4QDU5DE2nQ9nDuVTqobD4b8mGGyPMbIZnqyMsEcaGQy67XIw/Jw==" + echo "COSMOS_EMULATOR_AVAILABLE=true" >> $env:GITHUB_ENV + + - name: Run Unit Tests shell: bash run: | export UT_PROJECTS=$(find ./dotnet -type f -name "*.UnitTests.csproj" | tr '\n' ' ') @@ -133,12 +144,20 @@ jobs: # Check if the project supports the target framework if [[ "$target_frameworks" == *"${{ matrix.targetFramework }}"* ]]; then - dotnet test -f ${{ matrix.targetFramework }} -c ${{ matrix.configuration }} $project --no-build -v Normal --logger trx --collect:"XPlat Code Coverage" --results-directory:"TestResults/Coverage/" -- DataCollectionRunSettings.DataCollectors.DataCollector.Configuration.ExcludeByAttribute=GeneratedCodeAttribute,CompilerGeneratedAttribute,ExcludeFromCodeCoverageAttribute + if [[ "${{ matrix.targetFramework }}" == "${{ env.COVERAGE_FRAMEWORK }}" ]]; then + dotnet test -f ${{ matrix.targetFramework }} -c ${{ matrix.configuration }} $project --no-build -v Normal --logger trx --collect:"XPlat Code Coverage" --results-directory:"TestResults/Coverage/" -- DataCollectionRunSettings.DataCollectors.DataCollector.Configuration.ExcludeByAttribute=GeneratedCodeAttribute,CompilerGeneratedAttribute,ExcludeFromCodeCoverageAttribute + else + dotnet test -f ${{ matrix.targetFramework }} -c ${{ matrix.configuration }} $project --no-build -v Normal --logger trx + fi else echo "Skipping $project - does not support target framework ${{ matrix.targetFramework }} (supports: $target_frameworks)" fi done - + env: + # Cosmos DB Emulator connection settings + COSMOSDB_ENDPOINT: https://localhost:8081 + COSMOSDB_KEY: C2y6yDjf5/R+ob0N8A7Cgv30VRDJIWEHLM+4QDU5DE2nQ9nDuVTqobD4b8mGGyPMbIZnqyMsEcaGQy67XIw/Jw== + - name: Log event name and matrix integration-tests shell: bash run: echo "github.event_name:${{ github.event_name }} matrix.integration-tests:${{ matrix.integration-tests }} github.event.action:${{ github.event.action }} github.event.pull_request.merged:${{ github.event.pull_request.merged }}" @@ -151,6 +170,14 @@ jobs: tenant-id: ${{ secrets.AZURE_TENANT_ID }} subscription-id: ${{ secrets.AZURE_SUBSCRIPTION_ID }} + # This setup action is required for both Durable Task and Azure Functions integration tests. + # We only run it on Ubuntu since the Durable Task and Azure Functions features are not available + # on .NET Framework (net472) which is what we use the Windows runner for. + - name: Set up Durable Task and Azure Functions Integration Test Emulators + if: github.event_name != 'pull_request' && matrix.integration-tests && matrix.os == 'ubuntu-latest' + uses: ./.github/actions/azure-functions-integration-setup + id: azure-functions-setup + - name: Run Integration Tests shell: bash if: github.event_name != 'pull_request' && matrix.integration-tests @@ -168,10 +195,16 @@ jobs: fi done env: + # Cosmos DB Emulator connection settings + COSMOSDB_ENDPOINT: https://localhost:8081 + COSMOSDB_KEY: C2y6yDjf5/R+ob0N8A7Cgv30VRDJIWEHLM+4QDU5DE2nQ9nDuVTqobD4b8mGGyPMbIZnqyMsEcaGQy67XIw/Jw== # OpenAI Models OpenAI__ApiKey: ${{ secrets.OPENAI__APIKEY }} OpenAI__ChatModelId: ${{ vars.OPENAI__CHATMODELID }} OpenAI__ChatReasoningModelId: ${{ vars.OPENAI__CHATREASONINGMODELID }} + # Azure OpenAI Models + AZURE_OPENAI_CHAT_DEPLOYMENT_NAME: ${{ vars.AZUREOPENAI__CHATDEPLOYMENTNAME }} + AZURE_OPENAI_ENDPOINT: ${{ vars.AZUREOPENAI__ENDPOINT }} # Azure AI Foundry AzureAI__Endpoint: ${{ secrets.AZUREAI__ENDPOINT }} AzureAI__DeploymentName: ${{ vars.AZUREAI__DEPLOYMENTNAME }} @@ -183,19 +216,22 @@ jobs: # Generate test reports and check coverage - name: Generate test reports - uses: danielpalme/ReportGenerator-GitHub-Action@5.4.18 + if: matrix.targetFramework == env.COVERAGE_FRAMEWORK + uses: danielpalme/ReportGenerator-GitHub-Action@5.5.1 with: reports: "./TestResults/Coverage/**/coverage.cobertura.xml" targetdir: "./TestResults/Reports" reporttypes: "HtmlInline;JsonSummary" - name: Upload coverage report artifact + if: matrix.targetFramework == env.COVERAGE_FRAMEWORK uses: actions/upload-artifact@v5 with: name: CoverageReport-${{ matrix.os }}-${{ matrix.targetFramework }}-${{ matrix.configuration }} # Artifact name path: ./TestResults/Reports # Directory containing files to upload - name: Check coverage + if: matrix.targetFramework == env.COVERAGE_FRAMEWORK shell: pwsh run: .github/workflows/dotnet-check-coverage.ps1 -JsonReportPath "TestResults/Reports/Summary.json" -CoverageThreshold $env:COVERAGE_THRESHOLD diff --git a/.github/workflows/dotnet-format.yml b/.github/workflows/dotnet-format.yml index a9fe090013..8d7c9febb7 100644 --- a/.github/workflows/dotnet-format.yml +++ b/.github/workflows/dotnet-format.yml @@ -22,7 +22,7 @@ jobs: fail-fast: false matrix: include: - - { dotnet: "9.0", configuration: Release, os: ubuntu-latest } + - { dotnet: "10.0", configuration: Release, os: ubuntu-latest } runs-on: ${{ matrix.os }} env: @@ -30,7 +30,7 @@ jobs: steps: - name: Check out code - uses: actions/checkout@v5 + uses: actions/checkout@v6 with: fetch-depth: 0 persist-credentials: false diff --git a/.github/workflows/markdown-link-check.yml b/.github/workflows/markdown-link-check.yml index 3b015fc6af..5c984c5796 100644 --- a/.github/workflows/markdown-link-check.yml +++ b/.github/workflows/markdown-link-check.yml @@ -19,7 +19,7 @@ jobs: runs-on: ubuntu-22.04 # check out the latest version of the code steps: - - uses: actions/checkout@v5 + - uses: actions/checkout@v6 with: persist-credentials: false diff --git a/.github/workflows/python-code-quality.yml b/.github/workflows/python-code-quality.yml index 871436509c..176eb3db99 100644 --- a/.github/workflows/python-code-quality.yml +++ b/.github/workflows/python-code-quality.yml @@ -27,7 +27,9 @@ jobs: env: UV_PYTHON: ${{ matrix.python-version }} steps: - - uses: actions/checkout@v5 + - uses: actions/checkout@v6 + with: + fetch-depth: 0 - name: Set up python and install the project id: python-setup uses: ./.github/actions/python-setup @@ -46,4 +48,6 @@ jobs: with: extra_args: --config python/.pre-commit-config.yaml --all-files - name: Run Mypy - run: uv run poe mypy + env: + GITHUB_BASE_REF: ${{ github.event.pull_request.base.ref || github.base_ref || 'main' }} + run: uv run poe ci-mypy diff --git a/.github/workflows/python-docs.yml b/.github/workflows/python-docs.yml index b2be4b6ad0..f962ec318f 100644 --- a/.github/workflows/python-docs.yml +++ b/.github/workflows/python-docs.yml @@ -24,7 +24,7 @@ jobs: run: working-directory: python steps: - - uses: actions/checkout@v5 + - uses: actions/checkout@v6 - name: Set up uv uses: astral-sh/setup-uv@v7 with: diff --git a/.github/workflows/python-lab-tests.yml b/.github/workflows/python-lab-tests.yml index ae526cf962..f5cb504d04 100644 --- a/.github/workflows/python-lab-tests.yml +++ b/.github/workflows/python-lab-tests.yml @@ -24,7 +24,7 @@ jobs: outputs: pythonChanges: ${{ steps.filter.outputs.python}} steps: - - uses: actions/checkout@v5 + - uses: actions/checkout@v6 - uses: dorny/paths-filter@v3 id: filter with: @@ -59,7 +59,7 @@ jobs: run: working-directory: python steps: - - uses: actions/checkout@v5 + - uses: actions/checkout@v6 - name: Set up python and install the project id: python-setup diff --git a/.github/workflows/python-merge-tests.yml b/.github/workflows/python-merge-tests.yml index bd5768b968..66b9122726 100644 --- a/.github/workflows/python-merge-tests.yml +++ b/.github/workflows/python-merge-tests.yml @@ -28,7 +28,7 @@ jobs: outputs: pythonChanges: ${{ steps.filter.outputs.python}} steps: - - uses: actions/checkout@v5 + - uses: actions/checkout@v6 - uses: dorny/paths-filter@v3 id: filter with: @@ -66,11 +66,16 @@ jobs: AZURE_OPENAI_RESPONSES_DEPLOYMENT_NAME: ${{ vars.AZUREOPENAI__RESPONSESDEPLOYMENTNAME }} AZURE_OPENAI_ENDPOINT: ${{ vars.AZUREOPENAI__ENDPOINT }} LOCAL_MCP_URL: ${{ vars.LOCAL_MCP__URL }} + # For Azure Functions integration tests + FUNCTIONS_WORKER_RUNTIME: "python" + DURABLE_TASK_SCHEDULER_CONNECTION_STRING: "Endpoint=http://localhost:8080;TaskHub=default;Authentication=None" + AzureWebJobsStorage: "UseDevelopmentStorage=true" + defaults: run: working-directory: python steps: - - uses: actions/checkout@v5 + - uses: actions/checkout@v6 - name: Set up python and install the project id: python-setup uses: ./.github/actions/python-setup @@ -87,6 +92,9 @@ jobs: client-id: ${{ secrets.AZURE_CLIENT_ID }} tenant-id: ${{ secrets.AZURE_TENANT_ID }} subscription-id: ${{ secrets.AZURE_SUBSCRIPTION_ID }} + - name: Set up Azure Functions Integration Test Emulators + uses: ./.github/actions/azure-functions-integration-setup + id: azure-functions-setup - name: Test with pytest timeout-minutes: 10 run: uv run poe all-tests -n logical --dist loadfile --dist worksteal --timeout 300 --retries 3 --retry-delay 10 @@ -127,7 +135,7 @@ jobs: run: working-directory: python steps: - - uses: actions/checkout@v5 + - uses: actions/checkout@v6 - name: Set up python and install the project id: python-setup uses: ./.github/actions/python-setup diff --git a/.github/workflows/python-release.yml b/.github/workflows/python-release.yml index 97f1ef2481..ba6e3689b0 100644 --- a/.github/workflows/python-release.yml +++ b/.github/workflows/python-release.yml @@ -23,7 +23,7 @@ jobs: run: working-directory: python steps: - - uses: actions/checkout@v5 + - uses: actions/checkout@v6 - name: Set up python and install the project id: python-setup uses: ./.github/actions/python-setup diff --git a/.github/workflows/python-test-coverage-report.yml b/.github/workflows/python-test-coverage-report.yml index 9ea5b8022d..fa36073fc6 100644 --- a/.github/workflows/python-test-coverage-report.yml +++ b/.github/workflows/python-test-coverage-report.yml @@ -19,7 +19,7 @@ jobs: run: working-directory: python steps: - - uses: actions/checkout@v5 + - uses: actions/checkout@v6 - name: Download coverage report uses: actions/download-artifact@v6 with: @@ -39,7 +39,7 @@ jobs: echo "PR_NUMBER=$PR_NUMBER" >> $GITHUB_ENV - name: Pytest coverage comment id: coverageComment - uses: MishaKav/pytest-coverage-comment@v1.1.57 + uses: MishaKav/pytest-coverage-comment@v1.2.0 with: github-token: ${{ secrets.GH_ACTIONS_PR_WRITE }} issue-number: ${{ env.PR_NUMBER }} diff --git a/.github/workflows/python-test-coverage.yml b/.github/workflows/python-test-coverage.yml index dd260ba5f6..6268e7d47d 100644 --- a/.github/workflows/python-test-coverage.yml +++ b/.github/workflows/python-test-coverage.yml @@ -20,7 +20,7 @@ jobs: env: UV_PYTHON: "3.10" steps: - - uses: actions/checkout@v5 + - uses: actions/checkout@v6 # Save the PR number to a file since the workflow_run event # in the coverage report workflow does not have access to it - name: Save PR number diff --git a/.github/workflows/python-tests.yml b/.github/workflows/python-tests.yml index 697a8ff4a7..07b9200a46 100644 --- a/.github/workflows/python-tests.yml +++ b/.github/workflows/python-tests.yml @@ -27,7 +27,7 @@ jobs: run: working-directory: python steps: - - uses: actions/checkout@v5 + - uses: actions/checkout@v6 - name: Set up python and install the project id: python-setup uses: ./.github/actions/python-setup diff --git a/.gitignore b/.gitignore index 70c1563f5a..e0c0691dc1 100644 --- a/.gitignore +++ b/.gitignore @@ -204,6 +204,20 @@ agents.md # AI .claude/ WARP.md +**/memory-bank/ +**/projectBrief.md + +# Azurite storage emulator files +*/__azurite_db_blob__.json +*/__azurite_db_blob_extent__.json +*/__azurite_db_queue__.json +*/__azurite_db_queue_extent__.json +*/__azurite_db_table__.json +*/__blobstorage__/ +*/__queuestorage__/ + +# Azure Functions local settings +local.settings.json # Frontend **/frontend/node_modules/ @@ -211,4 +225,15 @@ WARP.md **/frontend/dist/ # Database files -*.db \ No newline at end of file +*.db + +# Package development docs (internal use only) +**/GAP_ANALYSIS.md +**/PR*_CHECKLIST.md +**/IMPLEMENTATION_NOTES.md + +# Development/local testing files +**/test_local.py +**/test_simple.py +**/test_streaming.py +**/test_sdk_functions.py diff --git a/README.md b/README.md index 30d9ab2bdd..64b0dbd821 100644 --- a/README.md +++ b/README.md @@ -48,6 +48,8 @@ dotnet add package Microsoft.Agents.AI - **[Migration from Semantic Kernel](https://learn.microsoft.com/en-us/agent-framework/migration-guide/from-semantic-kernel)** - Guide to migrate from Semantic Kernel - **[Migration from AutoGen](https://learn.microsoft.com/en-us/agent-framework/migration-guide/from-autogen)** - Guide to migrate from AutoGen +Still have questions? Join our [weekly office hours](./COMMUNITY.md#public-community-office-hours) or ask questions in our [Discord channel](https://discord.gg/b5zjErwbQM) to get help from the team and other users. + ### ✨ **Highlights** - **Graph-based Workflows**: Connect agents and deterministic functions using data flows with streaming, checkpointing, human-in-the-loop, and time-travel capabilities diff --git a/TRANSPARENCY_FAQ.md b/TRANSPARENCY_FAQ.md index cd850ff796..3a09f191eb 100644 --- a/TRANSPARENCY_FAQ.md +++ b/TRANSPARENCY_FAQ.md @@ -42,9 +42,9 @@ Microsoft Agent Framework relies on existing LLMs. Using the framework retains c **Framework-Specific Limitations**: -- **Platform Requirements**: Python 3.10+ required, specific .NET versions (.NET 8.0, 9.0, netstandard2.0, net472) +- **Platform Requirements**: Python 3.10+ required, specific .NET versions (.NET 8.0, 9.0, 10.0, netstandard2.0, net472) - **API Dependencies**: Requires proper configuration of LLM provider keys and endpoints -- **Orchestration Features**: Advanced orchestration patterns like GroupChat, Sequential, and Concurrent orchestrations are "coming soon" for Python implementation +- **Orchestration Features**: Advanced orchestration patterns including GroupChat, Sequential, and Concurrent workflows are now available in both Python and .NET implementations. See the respective language documentation for examples. - **Privacy and Data Protection**: The framework allows for human participation in conversations between agents. It is important to ensure that user data and conversations are protected and that developers use appropriate measures to safeguard privacy. - **Accountability and Transparency**: The framework involves multiple agents conversing and collaborating, it is important to establish clear accountability and transparency mechanisms. Users should be able to understand and trace the decision-making process of the agents involved in order to ensure accountability and address any potential issues or biases. - **Security & unintended consequences**: The use of multi-agent conversations and automation in complex tasks may have unintended consequences. Especially, allowing agents to make changes in external environments through tool calls or function execution could pose significant risks. Developers should carefully consider the potential risks and ensure that appropriate safeguards are in place to prevent harm or negative outcomes, including keeping a human in the loop for decision making. diff --git a/agent-samples/README.md b/agent-samples/README.md new file mode 100644 index 0000000000..0ee940f3a0 --- /dev/null +++ b/agent-samples/README.md @@ -0,0 +1,3 @@ +# Declarative Agents + +This folder contains sample agent definitions than be ran using the declarative agent support, for python see the [declarative agent python sample folder](../python/samples/getting_started/declarative/). diff --git a/agent-samples/azure/AzureOpenAI.yaml b/agent-samples/azure/AzureOpenAI.yaml new file mode 100644 index 0000000000..2f43d9ac92 --- /dev/null +++ b/agent-samples/azure/AzureOpenAI.yaml @@ -0,0 +1,25 @@ +kind: Prompt +name: Assistant +description: Helpful assistant +instructions: You are a helpful assistant. You answer questions is the language specified by the user. You return your answers in a JSON format. You must include Chat as the type in your response. +model: + id: =Env.AZURE_OPENAI_DEPLOYMENT_NAME + provider: AzureOpenAI + apiType: Chat + options: + temperature: 0.9 + topP: 0.95 +outputSchema: + properties: + language: + kind: string + required: true + description: The language of the answer. + answer: + kind: string + required: true + description: The answer text. + type: + kind: string + required: true + description: The type of the response. diff --git a/agent-samples/azure/AzureOpenAIAssistants.yaml b/agent-samples/azure/AzureOpenAIAssistants.yaml new file mode 100644 index 0000000000..f973d05acc --- /dev/null +++ b/agent-samples/azure/AzureOpenAIAssistants.yaml @@ -0,0 +1,25 @@ +kind: Prompt +name: Assistant +description: Helpful assistant +instructions: You are a helpful assistant. You answer questions in the language specified by the user. You return your answers in a JSON format. You must include Assistants as the type in your response. +model: + id: gpt-4o-mini + provider: AzureOpenAI + apiType: Assistants + options: + temperature: 0.9 + topP: 0.95 +outputSchema: + properties: + language: + type: string + required: true + description: The language of the answer. + answer: + type: string + required: true + description: The answer text. + type: + type: string + required: true + description: The type of the response. diff --git a/agent-samples/azure/AzureOpenAIChat.yaml b/agent-samples/azure/AzureOpenAIChat.yaml new file mode 100644 index 0000000000..d02e0c6039 --- /dev/null +++ b/agent-samples/azure/AzureOpenAIChat.yaml @@ -0,0 +1,25 @@ +kind: Prompt +name: Assistant +description: Helpful assistant +instructions: You are a helpful assistant. You answer questions in the language specified by the user. You return your answers in a JSON format. You must include Chat as the type in your response. +model: + id: gpt-4o-mini + provider: AzureOpenAI + apiType: Chat + options: + temperature: 0.9 + topP: 0.95 +outputSchema: + properties: + language: + type: string + required: true + description: The language of the answer. + answer: + type: string + required: true + description: The answer text. + type: + type: string + required: true + description: The type of the response. diff --git a/agent-samples/azure/AzureOpenAIResponses.yaml b/agent-samples/azure/AzureOpenAIResponses.yaml new file mode 100644 index 0000000000..006c1476f4 --- /dev/null +++ b/agent-samples/azure/AzureOpenAIResponses.yaml @@ -0,0 +1,25 @@ +kind: Prompt +name: Assistant +description: Helpful assistant +instructions: You are a helpful assistant. You answer questions in the language specified by the user. You return your answers in a JSON format. You must include Responses as the type in your response. +model: + id: gpt-4o-mini + provider: AzureOpenAI + apiType: Responses + options: + temperature: 0.9 + topP: 0.95 +outputSchema: + properties: + language: + type: string + required: true + description: The language of the answer. + answer: + type: string + required: true + description: The answer text. + type: + type: string + required: true + description: The type of the response. diff --git a/agent-samples/chatclient/Assistant.yaml b/agent-samples/chatclient/Assistant.yaml new file mode 100644 index 0000000000..3332d54540 --- /dev/null +++ b/agent-samples/chatclient/Assistant.yaml @@ -0,0 +1,18 @@ +kind: Prompt +name: Assistant +description: Helpful assistant +instructions: You are a helpful assistant. You answer questions in the language specified by the user. You return your answers in a JSON format. +model: + options: + temperature: 0.9 + topP: 0.95 +outputSchema: + properties: + language: + type: string + required: true + description: The language of the answer. + answer: + type: string + required: true + description: The answer text. diff --git a/agent-samples/chatclient/GetWeather.yaml b/agent-samples/chatclient/GetWeather.yaml new file mode 100644 index 0000000000..f32411be98 --- /dev/null +++ b/agent-samples/chatclient/GetWeather.yaml @@ -0,0 +1,29 @@ +kind: Prompt +name: Assistant +description: Helpful assistant +instructions: You are a helpful assistant. You answer questions using the tools provided. +model: + options: + temperature: 0.9 + topP: 0.95 + allowMultipleToolCalls: true + chatToolMode: auto +tools: + - kind: function + name: GetWeather + description: Get the weather for a given location. + bindings: + get_weather: get_weather + parameters: + properties: + location: + kind: string + description: The city and state, e.g. San Francisco, CA + required: true + unit: + kind: string + description: The unit of temperature. Possible values are 'celsius' and 'fahrenheit'. + required: false + enum: + - celsius + - fahrenheit diff --git a/agent-samples/foundry/FoundryAgent.yaml b/agent-samples/foundry/FoundryAgent.yaml new file mode 100644 index 0000000000..2de2ea069e --- /dev/null +++ b/agent-samples/foundry/FoundryAgent.yaml @@ -0,0 +1,22 @@ +kind: Prompt +name: Assistant +description: Helpful assistant +instructions: You are a helpful assistant. You answer questions in the language specified by the user. You return your answers in a JSON format. +model: + id: gpt-4.1-mini + options: + temperature: 0.9 + topP: 0.95 + connection: + kind: Remote + endpoint: =Env.AZURE_FOUNDRY_PROJECT_ENDPOINT +outputSchema: + properties: + language: + type: string + required: true + description: The language of the answer. + answer: + type: string + required: true + description: The answer text. diff --git a/agent-samples/foundry/MicrosoftLearnAgent.yaml b/agent-samples/foundry/MicrosoftLearnAgent.yaml new file mode 100644 index 0000000000..8e15340351 --- /dev/null +++ b/agent-samples/foundry/MicrosoftLearnAgent.yaml @@ -0,0 +1,21 @@ +kind: Prompt +name: MicrosoftLearnAgent +description: Microsoft Learn Agent +instructions: You answer questions by searching the Microsoft Learn content only. +model: + id: =Env.AZURE_FOUNDRY_PROJECT_MODEL_ID + options: + temperature: 0.9 + topP: 0.95 + connection: + kind: remote + endpoint: =Env.AZURE_FOUNDRY_PROJECT_ENDPOINT +tools: + - kind: mcp + name: microsoft_learn + description: Get information from Microsoft Learn. + url: https://learn.microsoft.com/api/mcp + approvalMode: + kind: never + allowedTools: + - microsoft_docs_search diff --git a/agent-samples/foundry/PersistentAgent.yaml b/agent-samples/foundry/PersistentAgent.yaml new file mode 100644 index 0000000000..298ded2202 --- /dev/null +++ b/agent-samples/foundry/PersistentAgent.yaml @@ -0,0 +1,22 @@ +kind: Prompt +name: Assistant +description: Helpful assistant +instructions: You are a helpful assistant. You answer questions is the language specified by the user. You return your answers in a JSON format. +model: + id: =Env.AZURE_FOUNDRY_PROJECT_MODEL_ID + options: + temperature: 0.9 + topP: 0.95 + connection: + kind: remote + endpoint: =Env.AZURE_FOUNDRY_PROJECT_ENDPOINT +outputSchema: + properties: + language: + kind: string + required: true + description: The language of the answer. + answer: + kind: string + required: true + description: The answer text. diff --git a/agent-samples/openai/OpenAI.yaml b/agent-samples/openai/OpenAI.yaml new file mode 100644 index 0000000000..0e70188fd6 --- /dev/null +++ b/agent-samples/openai/OpenAI.yaml @@ -0,0 +1,28 @@ +kind: Prompt +name: Assistant +description: Helpful assistant +instructions: You are a helpful assistant. You answer questions is the language specified by the user. You return your answers in a JSON format. You must include Chat as the type in your response. +model: + id: =Env.OPENAI_MODEL + provider: OpenAI + apiType: Chat + options: + temperature: 0.9 + topP: 0.95 + connection: + kind: key + key: =Env.OPENAI_API_KEY +outputSchema: + properties: + language: + kind: string + required: true + description: The language of the answer. + answer: + kind: string + required: true + description: The answer text. + type: + kind: string + required: true + description: The type of the response. diff --git a/agent-samples/openai/OpenAIAssistants.yaml b/agent-samples/openai/OpenAIAssistants.yaml new file mode 100644 index 0000000000..1318051120 --- /dev/null +++ b/agent-samples/openai/OpenAIAssistants.yaml @@ -0,0 +1,28 @@ +kind: Prompt +name: Assistant +description: Helpful assistant +instructions: You are a helpful assistant. You answer questions in the language specified by the user. You return your answers in a JSON format. You must include Assistants as the type in your response. +model: + id: gpt-4.1-mini + provider: OpenAI + apiType: Assistants + options: + temperature: 0.9 + topP: 0.95 + connection: + kind: ApiKey + key: =Env.OPENAI_API_KEY +outputSchema: + properties: + language: + type: string + required: true + description: The language of the answer. + answer: + type: string + required: true + description: The answer text. + type: + type: string + required: true + description: The type of the response. diff --git a/agent-samples/openai/OpenAIChat.yaml b/agent-samples/openai/OpenAIChat.yaml new file mode 100644 index 0000000000..78286aea5c --- /dev/null +++ b/agent-samples/openai/OpenAIChat.yaml @@ -0,0 +1,28 @@ +kind: Prompt +name: Assistant +description: Helpful assistant +instructions: You are a helpful assistant. You answer questions in the language specified by the user. You return your answers in a JSON format. You must include Chat as the type in your response. +model: + id: gpt-4.1-mini + provider: OpenAI + apiType: Chat + options: + temperature: 0.9 + topP: 0.95 + connection: + kind: ApiKey + key: =Env.OPENAI_API_KEY +outputSchema: + properties: + language: + type: string + required: true + description: The language of the answer. + answer: + type: string + required: true + description: The answer text. + type: + type: string + required: true + description: The type of the response. diff --git a/agent-samples/openai/OpenAIResponses.yaml b/agent-samples/openai/OpenAIResponses.yaml new file mode 100644 index 0000000000..bdc04d4a13 --- /dev/null +++ b/agent-samples/openai/OpenAIResponses.yaml @@ -0,0 +1,28 @@ +kind: Prompt +name: Assistant +description: Helpful assistant +instructions: You are a helpful assistant. You answer questions in the language specified by the user. You return your answers in a JSON format. You must include Responses as the type in your response. +model: + id: gpt-4.1-mini + provider: OpenAI + apiType: Responses + options: + temperature: 0.9 + topP: 0.95 + connection: + kind: key + apiKey: =Env.OPENAI_APIKEY +outputSchema: + properties: + language: + kind: string + required: true + description: The language of the answer. + answer: + kind: string + required: true + description: The answer text. + type: + kind: string + required: true + description: The type of the response. diff --git a/docs/assets/Agentic-framework_high-res.png b/docs/assets/Agentic-framework_high-res.png new file mode 100644 index 0000000000..cdb53b11bf Binary files /dev/null and b/docs/assets/Agentic-framework_high-res.png differ diff --git a/dotnet/Directory.Build.props b/dotnet/Directory.Build.props index 6b61196bbd..2482c43013 100644 --- a/dotnet/Directory.Build.props +++ b/dotnet/Directory.Build.props @@ -3,17 +3,14 @@ true true - AllEnabledByDefault - latest + 10.0-all true - 13 + latest enable - $(NoWarn);NU5128 + $(NoWarn);NU5128;CS8002 true - net9.0;net8.0 - net9.0 - net9.0;net8.0;netstandard2.0;net472 - net9.0;net472 + net10.0;net9.0;net8.0 + $(TargetFrameworksCore);netstandard2.0;net472 true Debug;Release;Publish diff --git a/dotnet/Directory.Build.targets b/dotnet/Directory.Build.targets index 75033d16e3..5e62f1cef7 100644 --- a/dotnet/Directory.Build.targets +++ b/dotnet/Directory.Build.targets @@ -5,7 +5,7 @@ - + diff --git a/dotnet/Directory.Packages.props b/dotnet/Directory.Packages.props index d110ec4426..c7a051bf83 100644 --- a/dotnet/Directory.Packages.props +++ b/dotnet/Directory.Packages.props @@ -7,34 +7,46 @@ - 9.5.2 + 13.0.2 - + + + - + - - - + + + + + + + + + + + + - + - - - + + + - - - - - - + + + + + + + @@ -44,41 +56,45 @@ - + - - + + - - - - - - - - - - - - - - - - - - + + + + + + + + + + + + + + + + + + - - - - - - - - + + + + + + + + - + + + + + @@ -86,36 +102,49 @@ - + - + - - - - - - + + + + + + + + + + + + + + + + + + - + + + - - + - + all runtime; build; native; contentfiles; analyzers; buildtransitive @@ -135,17 +164,17 @@ all runtime; build; native; contentfiles; analyzers; buildtransitive - + all runtime; build; native; contentfiles; analyzers; buildtransitive - + all runtime; build; native; contentfiles; analyzers; buildtransitive - + all runtime; build; native; contentfiles; analyzers; buildtransitive diff --git a/dotnet/agent-framework-dotnet.slnx b/dotnet/agent-framework-dotnet.slnx index 03f8a910d3..5e08a766f9 100644 --- a/dotnet/agent-framework-dotnet.slnx +++ b/dotnet/agent-framework-dotnet.slnx @@ -20,23 +20,39 @@ + + + + + + + + + + + + + - + + + + @@ -47,8 +63,7 @@ - - + @@ -58,28 +73,93 @@ - + - - - + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + - + + + + + + + + + + + + + + + + + + + + + + @@ -105,13 +185,22 @@ + + + + + + + + + + - @@ -142,7 +231,7 @@ - + @@ -154,10 +243,13 @@ - - - - + + + + + + + @@ -258,6 +350,7 @@ + @@ -282,16 +375,24 @@ + + + + + + + + @@ -299,9 +400,13 @@ + + + + @@ -312,15 +417,23 @@ + - + + + + + + + + - + \ No newline at end of file diff --git a/dotnet/agent-framework-release.slnf b/dotnet/agent-framework-release.slnf new file mode 100644 index 0000000000..ed8ac19465 --- /dev/null +++ b/dotnet/agent-framework-release.slnf @@ -0,0 +1,31 @@ +{ + "solution": { + "path": "agent-framework-dotnet.slnx", + "projects": [ + "src\\Microsoft.Agents.AI.A2A\\Microsoft.Agents.AI.A2A.csproj", + "src\\Microsoft.Agents.AI.Abstractions\\Microsoft.Agents.AI.Abstractions.csproj", + "src\\Microsoft.Agents.AI.AGUI\\Microsoft.Agents.AI.AGUI.csproj", + "src\\Microsoft.Agents.AI.Anthropic\\Microsoft.Agents.AI.Anthropic.csproj", + "src\\Microsoft.Agents.AI.AzureAI.Persistent\\Microsoft.Agents.AI.AzureAI.Persistent.csproj", + "src\\Microsoft.Agents.AI.AzureAI\\Microsoft.Agents.AI.AzureAI.csproj", + "src\\Microsoft.Agents.AI.CopilotStudio\\Microsoft.Agents.AI.CopilotStudio.csproj", + "src\\Microsoft.Agents.AI.CosmosNoSql\\Microsoft.Agents.AI.CosmosNoSql.csproj", + "src\\Microsoft.Agents.AI.Declarative\\Microsoft.Agents.AI.Declarative.csproj", + "src\\Microsoft.Agents.AI.DevUI\\Microsoft.Agents.AI.DevUI.csproj", + "src\\Microsoft.Agents.AI.DurableTask\\Microsoft.Agents.AI.DurableTask.csproj", + "src\\Microsoft.Agents.AI.Hosting.A2A.AspNetCore\\Microsoft.Agents.AI.Hosting.A2A.AspNetCore.csproj", + "src\\Microsoft.Agents.AI.Hosting.A2A\\Microsoft.Agents.AI.Hosting.A2A.csproj", + "src\\Microsoft.Agents.AI.Hosting.AGUI.AspNetCore\\Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.csproj", + "src\\Microsoft.Agents.AI.Hosting.AzureFunctions\\Microsoft.Agents.AI.Hosting.AzureFunctions.csproj", + "src\\Microsoft.Agents.AI.Hosting.OpenAI\\Microsoft.Agents.AI.Hosting.OpenAI.csproj", + "src\\Microsoft.Agents.AI.Hosting\\Microsoft.Agents.AI.Hosting.csproj", + "src\\Microsoft.Agents.AI.Mem0\\Microsoft.Agents.AI.Mem0.csproj", + "src\\Microsoft.Agents.AI.OpenAI\\Microsoft.Agents.AI.OpenAI.csproj", + "src\\Microsoft.Agents.AI.Purview\\Microsoft.Agents.AI.Purview.csproj", + "src\\Microsoft.Agents.AI.Workflows.Declarative.AzureAI\\Microsoft.Agents.AI.Workflows.Declarative.AzureAI.csproj", + "src\\Microsoft.Agents.AI.Workflows.Declarative\\Microsoft.Agents.AI.Workflows.Declarative.csproj", + "src\\Microsoft.Agents.AI.Workflows\\Microsoft.Agents.AI.Workflows.csproj", + "src\\Microsoft.Agents.AI\\Microsoft.Agents.AI.csproj" + ] + } +} diff --git a/dotnet/eng/MSBuild/Shared.props b/dotnet/eng/MSBuild/Shared.props index 54f93699ad..da8806a1f3 100644 --- a/dotnet/eng/MSBuild/Shared.props +++ b/dotnet/eng/MSBuild/Shared.props @@ -11,4 +11,13 @@ + + + + + + + + + diff --git a/dotnet/global.json b/dotnet/global.json index 402d97f665..54533bf771 100644 --- a/dotnet/global.json +++ b/dotnet/global.json @@ -1,7 +1,7 @@ { "sdk": { - "version": "9.0.300", - "rollForward": "latestMajor", + "version": "10.0.100", + "rollForward": "minor", "allowPrerelease": false } } \ No newline at end of file diff --git a/dotnet/nuget/nuget-package.props b/dotnet/nuget/nuget-package.props index 69476cc713..6ae60933c6 100644 --- a/dotnet/nuget/nuget-package.props +++ b/dotnet/nuget/nuget-package.props @@ -2,9 +2,9 @@ 1.0.0 - $(VersionPrefix)-$(VersionSuffix).251105.1 - $(VersionPrefix)-preview.251105.1 - 1.0.0-preview.251105.1 + $(VersionPrefix)-$(VersionSuffix).251204.1 + $(VersionPrefix)-preview.251204.1 + 1.0.0-preview.251204.1 Debug;Release;Publish true diff --git a/dotnet/samples/.editorconfig b/dotnet/samples/.editorconfig index d260a0e568..6da078d7c5 100644 --- a/dotnet/samples/.editorconfig +++ b/dotnet/samples/.editorconfig @@ -1,6 +1,7 @@ # Suppressing errors for Sample projects under dotnet/samples folder [*.cs] dotnet_diagnostic.CA1716.severity = none # Add summary to documentation comment. +dotnet_diagnostic.CA1873.severity = none # Evaluation of logging arguments may be expensive dotnet_diagnostic.CA2000.severity = none # Call System.IDisposable.Dispose on object before all references to it are out of scope dotnet_diagnostic.CA2007.severity = none # Do not directly await a Task diff --git a/dotnet/samples/A2AClientServer/A2AClient/A2AClient.csproj b/dotnet/samples/A2AClientServer/A2AClient/A2AClient.csproj index 77a0588231..6b88c5c697 100644 --- a/dotnet/samples/A2AClientServer/A2AClient/A2AClient.csproj +++ b/dotnet/samples/A2AClientServer/A2AClient/A2AClient.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable 5ee045b0-aea3-4f08-8d31-32d1a6f8fed0 @@ -12,8 +12,6 @@ - - diff --git a/dotnet/samples/A2AClientServer/A2AClient/HostClientAgent.cs b/dotnet/samples/A2AClientServer/A2AClient/HostClientAgent.cs index 817e5084e6..5ebae80ffe 100644 --- a/dotnet/samples/A2AClientServer/A2AClient/HostClientAgent.cs +++ b/dotnet/samples/A2AClientServer/A2AClient/HostClientAgent.cs @@ -4,6 +4,7 @@ using Microsoft.Extensions.AI; using Microsoft.Extensions.Logging; using OpenAI; +using OpenAI.Chat; namespace A2A; diff --git a/dotnet/samples/A2AClientServer/A2AServer/A2AServer.csproj b/dotnet/samples/A2AClientServer/A2AServer/A2AServer.csproj index 8d67180f64..0a3b170a0b 100644 --- a/dotnet/samples/A2AClientServer/A2AServer/A2AServer.csproj +++ b/dotnet/samples/A2AClientServer/A2AServer/A2AServer.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable 5ee045b0-aea3-4f08-8d31-32d1a6f8fed0 @@ -11,8 +11,11 @@ - - + + + + + diff --git a/dotnet/samples/A2AClientServer/A2AServer/HostAgentFactory.cs b/dotnet/samples/A2AClientServer/A2AServer/HostAgentFactory.cs index 81fd24c595..9c4fbaaf2c 100644 --- a/dotnet/samples/A2AClientServer/A2AServer/HostAgentFactory.cs +++ b/dotnet/samples/A2AClientServer/A2AServer/HostAgentFactory.cs @@ -6,6 +6,7 @@ using Microsoft.Agents.AI; using Microsoft.Extensions.AI; using OpenAI; +using OpenAI.Chat; namespace A2AServer; diff --git a/dotnet/samples/A2AClientServer/README.md b/dotnet/samples/A2AClientServer/README.md index 8bf5fc5816..04b9968e76 100644 --- a/dotnet/samples/A2AClientServer/README.md +++ b/dotnet/samples/A2AClientServer/README.md @@ -103,7 +103,7 @@ dotnet run --urls "http://localhost:5002;https://localhost:5012" --agentId " Exe - net9.0 + net10.0 enable enable a8b2e9f0-1ea3-4f18-9d41-42d1a6f8fe10 @@ -11,11 +11,10 @@ - - + diff --git a/dotnet/samples/AGUIClientServer/AGUIClient/AGUIClientSerializerContext.cs b/dotnet/samples/AGUIClientServer/AGUIClient/AGUIClientSerializerContext.cs new file mode 100644 index 0000000000..1cc4fb8f53 --- /dev/null +++ b/dotnet/samples/AGUIClientServer/AGUIClient/AGUIClientSerializerContext.cs @@ -0,0 +1,12 @@ +// Copyright (c) Microsoft. All rights reserved. + +// This sample demonstrates how to use the AG-UI client to connect to a remote AG-UI server +// and display streaming updates including conversation/response metadata, text content, and errors. + +using System.Text.Json.Serialization; + +namespace AGUIClient; + +[JsonSerializable(typeof(SensorRequest))] +[JsonSerializable(typeof(SensorResponse))] +internal sealed partial class AGUIClientSerializerContext : JsonSerializerContext; diff --git a/dotnet/samples/AGUIClientServer/AGUIClient/Program.cs b/dotnet/samples/AGUIClientServer/AGUIClient/Program.cs index 0c6a6539a8..3079bf1451 100644 --- a/dotnet/samples/AGUIClientServer/AGUIClient/Program.cs +++ b/dotnet/samples/AGUIClientServer/AGUIClient/Program.cs @@ -4,7 +4,9 @@ // and display streaming updates including conversation/response metadata, text content, and errors. using System.CommandLine; +using System.ComponentModel; using System.Reflection; +using System.Text; using Microsoft.Agents.AI; using Microsoft.Agents.AI.AGUI; using Microsoft.Extensions.AI; @@ -51,11 +53,40 @@ private static async Task HandleCommandsAsync(CancellationToken cancellationToke Timeout = TimeSpan.FromSeconds(60) }; - AGUIAgent agent = new( - id: "agui-client", + var changeBackground = AIFunctionFactory.Create( + () => + { + Console.ForegroundColor = ConsoleColor.DarkBlue; + Console.WriteLine("Changing color to blue"); + }, + name: "change_background_color", + description: "Change the console background color to dark blue." + ); + + var readClientClimateSensors = AIFunctionFactory.Create( + ([Description("The sensors measurements to include in the response")] SensorRequest request) => + { + return new SensorResponse() + { + Temperature = 22.5, + Humidity = 45.0, + AirQualityIndex = 75 + }; + }, + name: "read_client_climate_sensors", + description: "Reads the climate sensor data from the client device.", + serializerOptions: AGUIClientSerializerContext.Default.Options + ); + + var chatClient = new AGUIChatClient( + httpClient, + serverUrl, + jsonSerializerOptions: AGUIClientSerializerContext.Default.Options); + + AIAgent agent = chatClient.CreateAIAgent( + name: "agui-client", description: "AG-UI Client Agent", - httpClient: httpClient, - endpoint: serverUrl); + tools: [changeBackground, readClientClimateSensors]); AgentThread thread = agent.GetNewThread(); List messages = [new(ChatRole.System, "You are a helpful assistant.")]; @@ -82,10 +113,12 @@ private static async Task HandleCommandsAsync(CancellationToken cancellationToke // Call RunStreamingAsync to get streaming updates bool isFirstUpdate = true; string? threadId = null; + var updates = new List(); await foreach (AgentRunResponseUpdate update in agent.RunStreamingAsync(messages, thread, cancellationToken: cancellationToken)) { // Use AsChatResponseUpdate to access ChatResponseUpdate properties ChatResponseUpdate chatUpdate = update.AsChatResponseUpdate(); + updates.Add(chatUpdate); if (chatUpdate.ConversationId != null) { threadId = chatUpdate.ConversationId; @@ -111,6 +144,25 @@ private static async Task HandleCommandsAsync(CancellationToken cancellationToke Console.ResetColor(); break; + case FunctionCallContent functionCallContent: + Console.ForegroundColor = ConsoleColor.Green; + Console.WriteLine($"\n[Function Call - Name: {functionCallContent.Name}, Arguments: {PrintArguments(functionCallContent.Arguments)}]"); + Console.ResetColor(); + break; + + case FunctionResultContent functionResultContent: + Console.ForegroundColor = ConsoleColor.Magenta; + if (functionResultContent.Exception != null) + { + Console.WriteLine($"\n[Function Result - Exception: {functionResultContent.Exception}]"); + } + else + { + Console.WriteLine($"\n[Function Result - Result: {functionResultContent.Result}]"); + } + Console.ResetColor(); + break; + case ErrorContent errorContent: Console.ForegroundColor = ConsoleColor.Red; string code = errorContent.AdditionalProperties?["Code"] as string ?? "Unknown"; @@ -120,6 +172,14 @@ private static async Task HandleCommandsAsync(CancellationToken cancellationToke } } } + if (updates.Count > 0 && !updates[^1].Contents.Any(c => c is TextContent)) + { + var lastUpdate = updates[^1]; + Console.ForegroundColor = ConsoleColor.Yellow; + Console.WriteLine(); + Console.WriteLine($"[Run Ended - Thread: {threadId}, Run: {lastUpdate.ResponseId}]"); + Console.ResetColor(); + } messages.Clear(); Console.WriteLine(); } @@ -134,4 +194,20 @@ private static async Task HandleCommandsAsync(CancellationToken cancellationToke return; } } + + private static string PrintArguments(IDictionary? arguments) + { + if (arguments == null) + { + return ""; + } + var builder = new StringBuilder().AppendLine(); + foreach (var kvp in arguments) + { + builder + .AppendLine($" Name: {kvp.Key}") + .AppendLine($" Value: {kvp.Value}"); + } + return builder.ToString(); + } } diff --git a/dotnet/samples/AGUIClientServer/AGUIClient/SensorRequest.cs b/dotnet/samples/AGUIClientServer/AGUIClient/SensorRequest.cs new file mode 100644 index 0000000000..76e6efa8de --- /dev/null +++ b/dotnet/samples/AGUIClientServer/AGUIClient/SensorRequest.cs @@ -0,0 +1,13 @@ +// Copyright (c) Microsoft. All rights reserved. + +// This sample demonstrates how to use the AG-UI client to connect to a remote AG-UI server +// and display streaming updates including conversation/response metadata, text content, and errors. + +namespace AGUIClient; + +internal sealed class SensorRequest +{ + public bool IncludeTemperature { get; set; } = true; + public bool IncludeHumidity { get; set; } = true; + public bool IncludeAirQualityIndex { get; set; } = true; +} diff --git a/dotnet/samples/AGUIClientServer/AGUIClient/SensorResponse.cs b/dotnet/samples/AGUIClientServer/AGUIClient/SensorResponse.cs new file mode 100644 index 0000000000..09ade6a0c7 --- /dev/null +++ b/dotnet/samples/AGUIClientServer/AGUIClient/SensorResponse.cs @@ -0,0 +1,13 @@ +// Copyright (c) Microsoft. All rights reserved. + +// This sample demonstrates how to use the AG-UI client to connect to a remote AG-UI server +// and display streaming updates including conversation/response metadata, text content, and errors. + +namespace AGUIClient; + +internal sealed class SensorResponse +{ + public double Temperature { get; set; } + public double Humidity { get; set; } + public int AirQualityIndex { get; set; } +} diff --git a/dotnet/samples/AGUIClientServer/AGUIDojoServer/AGUIDojoServer.csproj b/dotnet/samples/AGUIClientServer/AGUIDojoServer/AGUIDojoServer.csproj new file mode 100644 index 0000000000..cea8efff76 --- /dev/null +++ b/dotnet/samples/AGUIClientServer/AGUIDojoServer/AGUIDojoServer.csproj @@ -0,0 +1,22 @@ + + + + Exe + net10.0 + enable + enable + b9c3f1e1-2fb4-5g29-0e52-53e2b7g9gf21 + + + + + + + + + + + + + + diff --git a/dotnet/samples/AGUIClientServer/AGUIDojoServer/AGUIDojoServerSerializerContext.cs b/dotnet/samples/AGUIClientServer/AGUIDojoServer/AGUIDojoServerSerializerContext.cs new file mode 100644 index 0000000000..c60db0efd0 --- /dev/null +++ b/dotnet/samples/AGUIClientServer/AGUIDojoServer/AGUIDojoServerSerializerContext.cs @@ -0,0 +1,23 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; +using AGUIDojoServer.AgenticUI; +using AGUIDojoServer.BackendToolRendering; +using AGUIDojoServer.PredictiveStateUpdates; +using AGUIDojoServer.SharedState; + +namespace AGUIDojoServer; + +[JsonSerializable(typeof(WeatherInfo))] +[JsonSerializable(typeof(Recipe))] +[JsonSerializable(typeof(Ingredient))] +[JsonSerializable(typeof(RecipeResponse))] +[JsonSerializable(typeof(Plan))] +[JsonSerializable(typeof(Step))] +[JsonSerializable(typeof(StepStatus))] +[JsonSerializable(typeof(StepStatus?))] +[JsonSerializable(typeof(JsonPatchOperation))] +[JsonSerializable(typeof(List))] +[JsonSerializable(typeof(List))] +[JsonSerializable(typeof(DocumentState))] +internal sealed partial class AGUIDojoServerSerializerContext : JsonSerializerContext; diff --git a/dotnet/samples/AGUIClientServer/AGUIDojoServer/AgenticUI/AgenticPlanningTools.cs b/dotnet/samples/AGUIClientServer/AGUIDojoServer/AgenticUI/AgenticPlanningTools.cs new file mode 100644 index 0000000000..98fe96b442 --- /dev/null +++ b/dotnet/samples/AGUIClientServer/AGUIDojoServer/AgenticUI/AgenticPlanningTools.cs @@ -0,0 +1,52 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.ComponentModel; + +namespace AGUIDojoServer.AgenticUI; + +internal static class AgenticPlanningTools +{ + [Description("Create a plan with multiple steps.")] + public static Plan CreatePlan([Description("List of step descriptions to create the plan.")] List steps) + { + return new Plan + { + Steps = [.. steps.Select(s => new Step { Description = s, Status = StepStatus.Pending })] + }; + } + + [Description("Update a step in the plan with new description or status.")] + public static async Task> UpdatePlanStepAsync( + [Description("The index of the step to update.")] int index, + [Description("The new description for the step (optional).")] string? description = null, + [Description("The new status for the step (optional).")] StepStatus? status = null) + { + var changes = new List(); + + if (description is not null) + { + changes.Add(new JsonPatchOperation + { + Op = "replace", + Path = $"/steps/{index}/description", + Value = description + }); + } + + if (status.HasValue) + { + // Status must be lowercase to match AG-UI frontend expectations: "pending" or "completed" + string statusValue = status.Value == StepStatus.Pending ? "pending" : "completed"; + changes.Add(new JsonPatchOperation + { + Op = "replace", + Path = $"/steps/{index}/status", + Value = statusValue + }); + } + + await Task.Delay(1000); + + return changes; + } +} diff --git a/dotnet/samples/AGUIClientServer/AGUIDojoServer/AgenticUI/AgenticUIAgent.cs b/dotnet/samples/AGUIClientServer/AGUIDojoServer/AgenticUI/AgenticUIAgent.cs new file mode 100644 index 0000000000..05a7d86f15 --- /dev/null +++ b/dotnet/samples/AGUIClientServer/AGUIDojoServer/AgenticUI/AgenticUIAgent.cs @@ -0,0 +1,88 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Diagnostics.CodeAnalysis; +using System.Runtime.CompilerServices; +using System.Text.Json; +using Microsoft.Agents.AI; +using Microsoft.Extensions.AI; + +namespace AGUIDojoServer.AgenticUI; + +[SuppressMessage("Performance", "CA1812:Avoid uninstantiated internal classes", Justification = "Instantiated by ChatClientAgentFactory.CreateAgenticUI")] +internal sealed class AgenticUIAgent : DelegatingAIAgent +{ + private readonly JsonSerializerOptions _jsonSerializerOptions; + + public AgenticUIAgent(AIAgent innerAgent, JsonSerializerOptions jsonSerializerOptions) + : base(innerAgent) + { + this._jsonSerializerOptions = jsonSerializerOptions; + } + + public override Task RunAsync(IEnumerable messages, AgentThread? thread = null, AgentRunOptions? options = null, CancellationToken cancellationToken = default) + { + return this.RunStreamingAsync(messages, thread, options, cancellationToken).ToAgentRunResponseAsync(cancellationToken); + } + + public override async IAsyncEnumerable RunStreamingAsync( + IEnumerable messages, + AgentThread? thread = null, + AgentRunOptions? options = null, + [EnumeratorCancellation] CancellationToken cancellationToken = default) + { + // Track function calls that should trigger state events + var trackedFunctionCalls = new Dictionary(); + + await foreach (var update in this.InnerAgent.RunStreamingAsync(messages, thread, options, cancellationToken).ConfigureAwait(false)) + { + // Process contents: track function calls and emit state events for results + List stateEventsToEmit = new(); + foreach (var content in update.Contents) + { + if (content is FunctionCallContent callContent) + { + if (callContent.Name == "create_plan" || callContent.Name == "update_plan_step") + { + trackedFunctionCalls[callContent.CallId] = callContent; + break; + } + } + else if (content is FunctionResultContent resultContent) + { + // Check if this result matches a tracked function call + if (trackedFunctionCalls.TryGetValue(resultContent.CallId, out var matchedCall)) + { + var bytes = JsonSerializer.SerializeToUtf8Bytes((JsonElement)resultContent.Result!, this._jsonSerializerOptions); + + // Determine event type based on the function name + if (matchedCall.Name == "create_plan") + { + stateEventsToEmit.Add(new DataContent(bytes, "application/json")); + } + else if (matchedCall.Name == "update_plan_step") + { + stateEventsToEmit.Add(new DataContent(bytes, "application/json-patch+json")); + } + } + } + } + + yield return update; + + yield return new AgentRunResponseUpdate( + new ChatResponseUpdate(role: ChatRole.System, stateEventsToEmit) + { + MessageId = "delta_" + Guid.NewGuid().ToString("N"), + CreatedAt = update.CreatedAt, + ResponseId = update.ResponseId, + AuthorName = update.AuthorName, + Role = update.Role, + ContinuationToken = update.ContinuationToken, + AdditionalProperties = update.AdditionalProperties, + }) + { + AgentId = update.AgentId + }; + } + } +} diff --git a/dotnet/samples/AGUIClientServer/AGUIDojoServer/AgenticUI/JsonPatchOperation.cs b/dotnet/samples/AGUIClientServer/AGUIDojoServer/AgenticUI/JsonPatchOperation.cs new file mode 100644 index 0000000000..1cd8f5dcd2 --- /dev/null +++ b/dotnet/samples/AGUIClientServer/AGUIDojoServer/AgenticUI/JsonPatchOperation.cs @@ -0,0 +1,20 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +namespace AGUIDojoServer.AgenticUI; + +internal sealed class JsonPatchOperation +{ + [JsonPropertyName("op")] + public required string Op { get; set; } + + [JsonPropertyName("path")] + public required string Path { get; set; } + + [JsonPropertyName("value")] + public object? Value { get; set; } + + [JsonPropertyName("from")] + public string? From { get; set; } +} diff --git a/dotnet/samples/AGUIClientServer/AGUIDojoServer/AgenticUI/Plan.cs b/dotnet/samples/AGUIClientServer/AGUIDojoServer/AgenticUI/Plan.cs new file mode 100644 index 0000000000..a8ffcc6c37 --- /dev/null +++ b/dotnet/samples/AGUIClientServer/AGUIDojoServer/AgenticUI/Plan.cs @@ -0,0 +1,11 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +namespace AGUIDojoServer.AgenticUI; + +internal sealed class Plan +{ + [JsonPropertyName("steps")] + public List Steps { get; set; } = []; +} diff --git a/dotnet/samples/AGUIClientServer/AGUIDojoServer/AgenticUI/Step.cs b/dotnet/samples/AGUIClientServer/AGUIDojoServer/AgenticUI/Step.cs new file mode 100644 index 0000000000..26bc9860a5 --- /dev/null +++ b/dotnet/samples/AGUIClientServer/AGUIDojoServer/AgenticUI/Step.cs @@ -0,0 +1,14 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +namespace AGUIDojoServer.AgenticUI; + +internal sealed class Step +{ + [JsonPropertyName("description")] + public required string Description { get; set; } + + [JsonPropertyName("status")] + public StepStatus Status { get; set; } = StepStatus.Pending; +} diff --git a/dotnet/samples/AGUIClientServer/AGUIDojoServer/AgenticUI/StepStatus.cs b/dotnet/samples/AGUIClientServer/AGUIDojoServer/AgenticUI/StepStatus.cs new file mode 100644 index 0000000000..f88d71bef0 --- /dev/null +++ b/dotnet/samples/AGUIClientServer/AGUIDojoServer/AgenticUI/StepStatus.cs @@ -0,0 +1,12 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +namespace AGUIDojoServer.AgenticUI; + +[JsonConverter(typeof(JsonStringEnumConverter))] +internal enum StepStatus +{ + Pending, + Completed +} diff --git a/dotnet/samples/AGUIClientServer/AGUIDojoServer/BackendToolRendering/WeatherInfo.cs b/dotnet/samples/AGUIClientServer/AGUIDojoServer/BackendToolRendering/WeatherInfo.cs new file mode 100644 index 0000000000..d6e3be9b80 --- /dev/null +++ b/dotnet/samples/AGUIClientServer/AGUIDojoServer/BackendToolRendering/WeatherInfo.cs @@ -0,0 +1,23 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +namespace AGUIDojoServer.BackendToolRendering; + +internal sealed class WeatherInfo +{ + [JsonPropertyName("temperature")] + public int Temperature { get; init; } + + [JsonPropertyName("conditions")] + public string Conditions { get; init; } = string.Empty; + + [JsonPropertyName("humidity")] + public int Humidity { get; init; } + + [JsonPropertyName("wind_speed")] + public int WindSpeed { get; init; } + + [JsonPropertyName("feelsLike")] + public int FeelsLike { get; init; } +} diff --git a/dotnet/samples/AGUIClientServer/AGUIDojoServer/ChatClientAgentFactory.cs b/dotnet/samples/AGUIClientServer/AGUIDojoServer/ChatClientAgentFactory.cs new file mode 100644 index 0000000000..58f5ad4ae9 --- /dev/null +++ b/dotnet/samples/AGUIClientServer/AGUIDojoServer/ChatClientAgentFactory.cs @@ -0,0 +1,180 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.ComponentModel; +using System.Text.Json; +using AGUIDojoServer.AgenticUI; +using AGUIDojoServer.BackendToolRendering; +using AGUIDojoServer.PredictiveStateUpdates; +using AGUIDojoServer.SharedState; +using Azure.AI.OpenAI; +using Azure.Identity; +using Microsoft.Agents.AI; +using Microsoft.Extensions.AI; +using ChatClient = OpenAI.Chat.ChatClient; + +namespace AGUIDojoServer; + +internal static class ChatClientAgentFactory +{ + private static AzureOpenAIClient? s_azureOpenAIClient; + private static string? s_deploymentName; + + public static void Initialize(IConfiguration configuration) + { + string endpoint = configuration["AZURE_OPENAI_ENDPOINT"] ?? throw new InvalidOperationException("AZURE_OPENAI_ENDPOINT is not set."); + s_deploymentName = configuration["AZURE_OPENAI_DEPLOYMENT_NAME"] ?? throw new InvalidOperationException("AZURE_OPENAI_DEPLOYMENT_NAME is not set."); + + s_azureOpenAIClient = new AzureOpenAIClient( + new Uri(endpoint), + new DefaultAzureCredential()); + } + + public static ChatClientAgent CreateAgenticChat() + { + ChatClient chatClient = s_azureOpenAIClient!.GetChatClient(s_deploymentName!); + + return chatClient.AsIChatClient().CreateAIAgent( + name: "AgenticChat", + description: "A simple chat agent using Azure OpenAI"); + } + + public static ChatClientAgent CreateBackendToolRendering() + { + ChatClient chatClient = s_azureOpenAIClient!.GetChatClient(s_deploymentName!); + + return chatClient.AsIChatClient().CreateAIAgent( + name: "BackendToolRenderer", + description: "An agent that can render backend tools using Azure OpenAI", + tools: [AIFunctionFactory.Create( + GetWeather, + name: "get_weather", + description: "Get the weather for a given location.", + AGUIDojoServerSerializerContext.Default.Options)]); + } + + public static ChatClientAgent CreateHumanInTheLoop() + { + ChatClient chatClient = s_azureOpenAIClient!.GetChatClient(s_deploymentName!); + + return chatClient.AsIChatClient().CreateAIAgent( + name: "HumanInTheLoopAgent", + description: "An agent that involves human feedback in its decision-making process using Azure OpenAI"); + } + + public static ChatClientAgent CreateToolBasedGenerativeUI() + { + ChatClient chatClient = s_azureOpenAIClient!.GetChatClient(s_deploymentName!); + + return chatClient.AsIChatClient().CreateAIAgent( + name: "ToolBasedGenerativeUIAgent", + description: "An agent that uses tools to generate user interfaces using Azure OpenAI"); + } + + public static AIAgent CreateAgenticUI(JsonSerializerOptions options) + { + ChatClient chatClient = s_azureOpenAIClient!.GetChatClient(s_deploymentName!); + var baseAgent = chatClient.AsIChatClient().CreateAIAgent(new ChatClientAgentOptions + { + Name = "AgenticUIAgent", + Description = "An agent that generates agentic user interfaces using Azure OpenAI", + ChatOptions = new ChatOptions + { + Instructions = """ + When planning use tools only, without any other messages. + IMPORTANT: + - Use the `create_plan` tool to set the initial state of the steps + - Use the `update_plan_step` tool to update the status of each step + - Do NOT repeat the plan or summarise it in a message + - Do NOT confirm the creation or updates in a message + - Do NOT ask the user for additional information or next steps + - Do NOT leave a plan hanging, always complete the plan via `update_plan_step` if one is ongoing. + - Continue calling update_plan_step until all steps are marked as completed. + + Only one plan can be active at a time, so do not call the `create_plan` tool + again until all the steps in current plan are completed. + """, + Tools = [ + AIFunctionFactory.Create( + AgenticPlanningTools.CreatePlan, + name: "create_plan", + description: "Create a plan with multiple steps.", + AGUIDojoServerSerializerContext.Default.Options), + AIFunctionFactory.Create( + AgenticPlanningTools.UpdatePlanStepAsync, + name: "update_plan_step", + description: "Update a step in the plan with new description or status.", + AGUIDojoServerSerializerContext.Default.Options) + ], + AllowMultipleToolCalls = false + } + }); + + return new AgenticUIAgent(baseAgent, options); + } + + public static AIAgent CreateSharedState(JsonSerializerOptions options) + { + ChatClient chatClient = s_azureOpenAIClient!.GetChatClient(s_deploymentName!); + + var baseAgent = chatClient.AsIChatClient().CreateAIAgent( + name: "SharedStateAgent", + description: "An agent that demonstrates shared state patterns using Azure OpenAI"); + + return new SharedStateAgent(baseAgent, options); + } + + public static AIAgent CreatePredictiveStateUpdates(JsonSerializerOptions options) + { + ChatClient chatClient = s_azureOpenAIClient!.GetChatClient(s_deploymentName!); + + var baseAgent = chatClient.AsIChatClient().CreateAIAgent(new ChatClientAgentOptions + { + Name = "PredictiveStateUpdatesAgent", + Description = "An agent that demonstrates predictive state updates using Azure OpenAI", + ChatOptions = new ChatOptions + { + Instructions = """ + You are a document editor assistant. When asked to write or edit content: + + IMPORTANT: + - Use the `write_document` tool with the full document text in Markdown format + - Format the document extensively so it's easy to read + - You can use all kinds of markdown (headings, lists, bold, etc.) + - However, do NOT use italic or strike-through formatting + - You MUST write the full document, even when changing only a few words + - When making edits to the document, try to make them minimal - do not change every word + - Keep stories SHORT! + - After you are done writing the document you MUST call a confirm_changes tool after you call write_document + + After the user confirms the changes, provide a brief summary of what you wrote. + """, + Tools = [ + AIFunctionFactory.Create( + WriteDocument, + name: "write_document", + description: "Write a document. Use markdown formatting to format the document.", + AGUIDojoServerSerializerContext.Default.Options) + ] + } + }); + + return new PredictiveStateUpdatesAgent(baseAgent, options); + } + + [Description("Get the weather for a given location.")] + private static WeatherInfo GetWeather([Description("The location to get the weather for.")] string location) => new() + { + Temperature = 20, + Conditions = "sunny", + Humidity = 50, + WindSpeed = 10, + FeelsLike = 25 + }; + + [Description("Write a document in markdown format.")] + private static string WriteDocument([Description("The document content to write.")] string document) + { + // Simply return success - the document is tracked via state updates + return "Document written successfully"; + } +} diff --git a/dotnet/samples/AGUIClientServer/AGUIDojoServer/PredictiveStateUpdates/DocumentState.cs b/dotnet/samples/AGUIClientServer/AGUIDojoServer/PredictiveStateUpdates/DocumentState.cs new file mode 100644 index 0000000000..ad053fe4a2 --- /dev/null +++ b/dotnet/samples/AGUIClientServer/AGUIDojoServer/PredictiveStateUpdates/DocumentState.cs @@ -0,0 +1,11 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +namespace AGUIDojoServer.PredictiveStateUpdates; + +internal sealed class DocumentState +{ + [JsonPropertyName("document")] + public string Document { get; set; } = string.Empty; +} diff --git a/dotnet/samples/AGUIClientServer/AGUIDojoServer/PredictiveStateUpdates/PredictiveStateUpdatesAgent.cs b/dotnet/samples/AGUIClientServer/AGUIDojoServer/PredictiveStateUpdates/PredictiveStateUpdatesAgent.cs new file mode 100644 index 0000000000..8ac9928fbe --- /dev/null +++ b/dotnet/samples/AGUIClientServer/AGUIDojoServer/PredictiveStateUpdates/PredictiveStateUpdatesAgent.cs @@ -0,0 +1,104 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Diagnostics.CodeAnalysis; +using System.Runtime.CompilerServices; +using System.Text.Json; +using Microsoft.Agents.AI; +using Microsoft.Extensions.AI; + +namespace AGUIDojoServer.PredictiveStateUpdates; + +[SuppressMessage("Performance", "CA1812:Avoid uninstantiated internal classes", Justification = "Instantiated by ChatClientAgentFactory.CreatePredictiveStateUpdates")] +internal sealed class PredictiveStateUpdatesAgent : DelegatingAIAgent +{ + private readonly JsonSerializerOptions _jsonSerializerOptions; + private const int ChunkSize = 10; // Characters per chunk for streaming effect + + public PredictiveStateUpdatesAgent(AIAgent innerAgent, JsonSerializerOptions jsonSerializerOptions) + : base(innerAgent) + { + this._jsonSerializerOptions = jsonSerializerOptions; + } + + public override Task RunAsync(IEnumerable messages, AgentThread? thread = null, AgentRunOptions? options = null, CancellationToken cancellationToken = default) + { + return this.RunStreamingAsync(messages, thread, options, cancellationToken).ToAgentRunResponseAsync(cancellationToken); + } + + public override async IAsyncEnumerable RunStreamingAsync( + IEnumerable messages, + AgentThread? thread = null, + AgentRunOptions? options = null, + [EnumeratorCancellation] CancellationToken cancellationToken = default) + { + // Track the last emitted document state to avoid duplicates + string? lastEmittedDocument = null; + + await foreach (var update in this.InnerAgent.RunStreamingAsync(messages, thread, options, cancellationToken).ConfigureAwait(false)) + { + // Check if we're seeing a write_document tool call and emit predictive state + bool hasToolCall = false; + string? documentContent = null; + + foreach (var content in update.Contents) + { + if (content is FunctionCallContent callContent && callContent.Name == "write_document") + { + hasToolCall = true; + // Try to extract the document argument directly from the dictionary + if (callContent.Arguments?.TryGetValue("document", out var documentValue) == true) + { + documentContent = documentValue?.ToString(); + } + } + } + + // Always yield the original update first + yield return update; + + // If we got a complete tool call with document content, "fake" stream it in chunks + if (hasToolCall && documentContent != null && documentContent != lastEmittedDocument) + { + // Chunk the document content and emit progressive state updates + int startIndex = 0; + if (lastEmittedDocument != null && documentContent.StartsWith(lastEmittedDocument, StringComparison.Ordinal)) + { + // Only stream the new portion that was added + startIndex = lastEmittedDocument.Length; + } + + // Stream the document in chunks + for (int i = startIndex; i < documentContent.Length; i += ChunkSize) + { + int length = Math.Min(ChunkSize, documentContent.Length - i); + string chunk = documentContent.Substring(0, i + length); + + // Prepare predictive state update as DataContent + var stateUpdate = new DocumentState { Document = chunk }; + byte[] stateBytes = JsonSerializer.SerializeToUtf8Bytes( + stateUpdate, + this._jsonSerializerOptions.GetTypeInfo(typeof(DocumentState))); + + yield return new AgentRunResponseUpdate( + new ChatResponseUpdate(role: ChatRole.Assistant, [new DataContent(stateBytes, "application/json")]) + { + MessageId = "snapshot" + Guid.NewGuid().ToString("N"), + CreatedAt = update.CreatedAt, + ResponseId = update.ResponseId, + AdditionalProperties = update.AdditionalProperties, + AuthorName = update.AuthorName, + ContinuationToken = update.ContinuationToken, + }) + { + AgentId = update.AgentId + }; + + // Small delay to simulate streaming + await Task.Delay(50, cancellationToken).ConfigureAwait(false); + } + + lastEmittedDocument = documentContent; + } + } + } +} diff --git a/dotnet/samples/AGUIClientServer/AGUIDojoServer/Program.cs b/dotnet/samples/AGUIClientServer/AGUIDojoServer/Program.cs new file mode 100644 index 0000000000..e3b0020362 --- /dev/null +++ b/dotnet/samples/AGUIClientServer/AGUIDojoServer/Program.cs @@ -0,0 +1,47 @@ +// Copyright (c) Microsoft. All rights reserved. + +using AGUIDojoServer; +using Microsoft.Agents.AI.Hosting.AGUI.AspNetCore; +using Microsoft.AspNetCore.HttpLogging; +using Microsoft.Extensions.Options; + +WebApplicationBuilder builder = WebApplication.CreateBuilder(args); + +builder.Services.AddHttpLogging(logging => +{ + logging.LoggingFields = HttpLoggingFields.RequestPropertiesAndHeaders | HttpLoggingFields.RequestBody + | HttpLoggingFields.ResponsePropertiesAndHeaders | HttpLoggingFields.ResponseBody; + logging.RequestBodyLogLimit = int.MaxValue; + logging.ResponseBodyLogLimit = int.MaxValue; +}); + +builder.Services.AddHttpClient().AddLogging(); +builder.Services.ConfigureHttpJsonOptions(options => options.SerializerOptions.TypeInfoResolverChain.Add(AGUIDojoServerSerializerContext.Default)); +builder.Services.AddAGUI(); + +WebApplication app = builder.Build(); + +app.UseHttpLogging(); + +// Initialize the factory +ChatClientAgentFactory.Initialize(app.Configuration); + +// Map the AG-UI agent endpoints for different scenarios +app.MapAGUI("/agentic_chat", ChatClientAgentFactory.CreateAgenticChat()); + +app.MapAGUI("/backend_tool_rendering", ChatClientAgentFactory.CreateBackendToolRendering()); + +app.MapAGUI("/human_in_the_loop", ChatClientAgentFactory.CreateHumanInTheLoop()); + +app.MapAGUI("/tool_based_generative_ui", ChatClientAgentFactory.CreateToolBasedGenerativeUI()); + +var jsonOptions = app.Services.GetRequiredService>(); +app.MapAGUI("/agentic_generative_ui", ChatClientAgentFactory.CreateAgenticUI(jsonOptions.Value.SerializerOptions)); + +app.MapAGUI("/shared_state", ChatClientAgentFactory.CreateSharedState(jsonOptions.Value.SerializerOptions)); + +app.MapAGUI("/predictive_state_updates", ChatClientAgentFactory.CreatePredictiveStateUpdates(jsonOptions.Value.SerializerOptions)); + +await app.RunAsync(); + +public partial class Program; diff --git a/dotnet/samples/AGUIClientServer/AGUIDojoServer/Properties/launchSettings.json b/dotnet/samples/AGUIClientServer/AGUIDojoServer/Properties/launchSettings.json new file mode 100644 index 0000000000..d1c2dbfa92 --- /dev/null +++ b/dotnet/samples/AGUIClientServer/AGUIDojoServer/Properties/launchSettings.json @@ -0,0 +1,12 @@ +{ + "profiles": { + "AGUIDojoServer": { + "commandName": "Project", + "launchBrowser": true, + "environmentVariables": { + "ASPNETCORE_ENVIRONMENT": "Development" + }, + "applicationUrl": "http://localhost:5018" + } + } +} \ No newline at end of file diff --git a/dotnet/samples/AGUIClientServer/AGUIDojoServer/SharedState/Ingredient.cs b/dotnet/samples/AGUIClientServer/AGUIDojoServer/SharedState/Ingredient.cs new file mode 100644 index 0000000000..d56d88d958 --- /dev/null +++ b/dotnet/samples/AGUIClientServer/AGUIDojoServer/SharedState/Ingredient.cs @@ -0,0 +1,17 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +namespace AGUIDojoServer.SharedState; + +internal sealed class Ingredient +{ + [JsonPropertyName("icon")] + public string Icon { get; set; } = string.Empty; + + [JsonPropertyName("name")] + public string Name { get; set; } = string.Empty; + + [JsonPropertyName("amount")] + public string Amount { get; set; } = string.Empty; +} diff --git a/dotnet/samples/AGUIClientServer/AGUIDojoServer/SharedState/Recipe.cs b/dotnet/samples/AGUIClientServer/AGUIDojoServer/SharedState/Recipe.cs new file mode 100644 index 0000000000..a8485da839 --- /dev/null +++ b/dotnet/samples/AGUIClientServer/AGUIDojoServer/SharedState/Recipe.cs @@ -0,0 +1,26 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +namespace AGUIDojoServer.SharedState; + +internal sealed class Recipe +{ + [JsonPropertyName("title")] + public string Title { get; set; } = string.Empty; + + [JsonPropertyName("skill_level")] + public string SkillLevel { get; set; } = string.Empty; + + [JsonPropertyName("cooking_time")] + public string CookingTime { get; set; } = string.Empty; + + [JsonPropertyName("special_preferences")] + public List SpecialPreferences { get; set; } = []; + + [JsonPropertyName("ingredients")] + public List Ingredients { get; set; } = []; + + [JsonPropertyName("instructions")] + public List Instructions { get; set; } = []; +} diff --git a/dotnet/samples/AGUIClientServer/AGUIDojoServer/SharedState/RecipeResponse.cs b/dotnet/samples/AGUIClientServer/AGUIDojoServer/SharedState/RecipeResponse.cs new file mode 100644 index 0000000000..dadf3b7a2b --- /dev/null +++ b/dotnet/samples/AGUIClientServer/AGUIDojoServer/SharedState/RecipeResponse.cs @@ -0,0 +1,13 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +namespace AGUIDojoServer.SharedState; + +#pragma warning disable CA1812 // Used for the JsonSchema response format +internal sealed class RecipeResponse +#pragma warning restore CA1812 +{ + [JsonPropertyName("recipe")] + public Recipe Recipe { get; set; } = new(); +} diff --git a/dotnet/samples/AGUIClientServer/AGUIDojoServer/SharedState/SharedStateAgent.cs b/dotnet/samples/AGUIClientServer/AGUIDojoServer/SharedState/SharedStateAgent.cs new file mode 100644 index 0000000000..c10450fcfb --- /dev/null +++ b/dotnet/samples/AGUIClientServer/AGUIDojoServer/SharedState/SharedStateAgent.cs @@ -0,0 +1,106 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Diagnostics.CodeAnalysis; +using System.Runtime.CompilerServices; +using System.Text.Json; +using Microsoft.Agents.AI; +using Microsoft.Extensions.AI; + +namespace AGUIDojoServer.SharedState; + +[SuppressMessage("Performance", "CA1812:Avoid uninstantiated internal classes", Justification = "Instantiated by ChatClientAgentFactory.CreateSharedState")] +internal sealed class SharedStateAgent : DelegatingAIAgent +{ + private readonly JsonSerializerOptions _jsonSerializerOptions; + + public SharedStateAgent(AIAgent innerAgent, JsonSerializerOptions jsonSerializerOptions) + : base(innerAgent) + { + this._jsonSerializerOptions = jsonSerializerOptions; + } + + public override Task RunAsync(IEnumerable messages, AgentThread? thread = null, AgentRunOptions? options = null, CancellationToken cancellationToken = default) + { + return this.RunStreamingAsync(messages, thread, options, cancellationToken).ToAgentRunResponseAsync(cancellationToken); + } + + public override async IAsyncEnumerable RunStreamingAsync( + IEnumerable messages, + AgentThread? thread = null, + AgentRunOptions? options = null, + [EnumeratorCancellation] CancellationToken cancellationToken = default) + { + if (options is not ChatClientAgentRunOptions { ChatOptions.AdditionalProperties: { } properties } chatRunOptions || + !properties.TryGetValue("ag_ui_state", out JsonElement state)) + { + await foreach (var update in this.InnerAgent.RunStreamingAsync(messages, thread, options, cancellationToken).ConfigureAwait(false)) + { + yield return update; + } + yield break; + } + + var firstRunOptions = new ChatClientAgentRunOptions + { + ChatOptions = chatRunOptions.ChatOptions.Clone(), + AllowBackgroundResponses = chatRunOptions.AllowBackgroundResponses, + ContinuationToken = chatRunOptions.ContinuationToken, + ChatClientFactory = chatRunOptions.ChatClientFactory, + }; + + // Configure JSON schema response format for structured state output + firstRunOptions.ChatOptions.ResponseFormat = ChatResponseFormat.ForJsonSchema( + schemaName: "RecipeResponse", + schemaDescription: "A response containing a recipe with title, skill level, cooking time, preferences, ingredients, and instructions"); + + ChatMessage stateUpdateMessage = new( + ChatRole.System, + [ + new TextContent("Here is the current state in JSON format:"), + new TextContent(state.GetRawText()), + new TextContent("The new state is:") + ]); + + var firstRunMessages = messages.Append(stateUpdateMessage); + + var allUpdates = new List(); + await foreach (var update in this.InnerAgent.RunStreamingAsync(firstRunMessages, thread, firstRunOptions, cancellationToken).ConfigureAwait(false)) + { + allUpdates.Add(update); + + // Yield all non-text updates (tool calls, etc.) + bool hasNonTextContent = update.Contents.Any(c => c is not TextContent); + if (hasNonTextContent) + { + yield return update; + } + } + + var response = allUpdates.ToAgentRunResponse(); + + if (response.TryDeserialize(this._jsonSerializerOptions, out JsonElement stateSnapshot)) + { + byte[] stateBytes = JsonSerializer.SerializeToUtf8Bytes( + stateSnapshot, + this._jsonSerializerOptions.GetTypeInfo(typeof(JsonElement))); + yield return new AgentRunResponseUpdate + { + Contents = [new DataContent(stateBytes, "application/json")] + }; + } + else + { + yield break; + } + + var secondRunMessages = messages.Concat(response.Messages).Append( + new ChatMessage( + ChatRole.System, + [new TextContent("Please provide a concise summary of the state changes in at most two sentences.")])); + + await foreach (var update in this.InnerAgent.RunStreamingAsync(secondRunMessages, thread, options, cancellationToken).ConfigureAwait(false)) + { + yield return update; + } + } +} diff --git a/dotnet/samples/AGUIClientServer/AGUIDojoServer/appsettings.Development.json b/dotnet/samples/AGUIClientServer/AGUIDojoServer/appsettings.Development.json new file mode 100644 index 0000000000..3e805edef8 --- /dev/null +++ b/dotnet/samples/AGUIClientServer/AGUIDojoServer/appsettings.Development.json @@ -0,0 +1,9 @@ +{ + "Logging": { + "LogLevel": { + "Default": "Information", + "Microsoft.AspNetCore": "Warning", + "Microsoft.AspNetCore.HttpLogging.HttpLoggingMiddleware": "Information" + } + } +} diff --git a/dotnet/samples/AGUIClientServer/AGUIDojoServer/appsettings.json b/dotnet/samples/AGUIClientServer/AGUIDojoServer/appsettings.json new file mode 100644 index 0000000000..bb20fb69dd --- /dev/null +++ b/dotnet/samples/AGUIClientServer/AGUIDojoServer/appsettings.json @@ -0,0 +1,10 @@ +{ + "Logging": { + "LogLevel": { + "Default": "Information", + "Microsoft.AspNetCore": "Warning", + "Microsoft.AspNetCore.HttpLogging.HttpLoggingMiddleware": "Information" + } + }, + "AllowedHosts": "*" +} diff --git a/dotnet/samples/AGUIClientServer/AGUIServer/AGUIServer.csproj b/dotnet/samples/AGUIClientServer/AGUIServer/AGUIServer.csproj index c1bcd511da..ccfe22923a 100644 --- a/dotnet/samples/AGUIClientServer/AGUIServer/AGUIServer.csproj +++ b/dotnet/samples/AGUIClientServer/AGUIServer/AGUIServer.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable a8b2e9f0-1ea3-4f18-9d41-42d1a6f8fe10 @@ -11,8 +11,6 @@ - - diff --git a/dotnet/samples/AGUIClientServer/AGUIServer/AGUIServerSerializerContext.cs b/dotnet/samples/AGUIClientServer/AGUIServer/AGUIServerSerializerContext.cs new file mode 100644 index 0000000000..1ca6ad7bdc --- /dev/null +++ b/dotnet/samples/AGUIClientServer/AGUIServer/AGUIServerSerializerContext.cs @@ -0,0 +1,9 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +namespace AGUIServer; + +[JsonSerializable(typeof(ServerWeatherForecastRequest))] +[JsonSerializable(typeof(ServerWeatherForecastResponse))] +internal sealed partial class AGUIServerSerializerContext : JsonSerializerContext; diff --git a/dotnet/samples/AGUIClientServer/AGUIServer/Program.cs b/dotnet/samples/AGUIClientServer/AGUIServer/Program.cs index f26ace30a1..bcfd86e60d 100644 --- a/dotnet/samples/AGUIClientServer/AGUIServer/Program.cs +++ b/dotnet/samples/AGUIClientServer/AGUIServer/Program.cs @@ -1,24 +1,49 @@ // Copyright (c) Microsoft. All rights reserved. +using System.ComponentModel; +using AGUIServer; using Azure.AI.OpenAI; using Azure.Identity; using Microsoft.Agents.AI.Hosting.AGUI.AspNetCore; using Microsoft.Extensions.AI; -using OpenAI; +using OpenAI.Chat; WebApplicationBuilder builder = WebApplication.CreateBuilder(args); builder.Services.AddHttpClient().AddLogging(); +builder.Services.ConfigureHttpJsonOptions(options => options.SerializerOptions.TypeInfoResolverChain.Add(AGUIServerSerializerContext.Default)); +builder.Services.AddAGUI(); + WebApplication app = builder.Build(); string endpoint = builder.Configuration["AZURE_OPENAI_ENDPOINT"] ?? throw new InvalidOperationException("AZURE_OPENAI_ENDPOINT is not set."); string deploymentName = builder.Configuration["AZURE_OPENAI_DEPLOYMENT_NAME"] ?? throw new InvalidOperationException("AZURE_OPENAI_DEPLOYMENT_NAME is not set."); -// Create the AI agent +// Create the AI agent with tools var agent = new AzureOpenAIClient( new Uri(endpoint), new DefaultAzureCredential()) .GetChatClient(deploymentName) - .CreateAIAgent(name: "AGUIAssistant"); + .CreateAIAgent( + name: "AGUIAssistant", + tools: [ + AIFunctionFactory.Create( + () => DateTimeOffset.UtcNow, + name: "get_current_time", + description: "Get the current UTC time." + ), + AIFunctionFactory.Create( + ([Description("The weather forecast request")]ServerWeatherForecastRequest request) => { + return new ServerWeatherForecastResponse() + { + Summary = "Sunny", + TemperatureC = 25, + Date = request.Date + }; + }, + name: "get_server_weather_forecast", + description: "Gets the forecast for a specific location and date", + AGUIServerSerializerContext.Default.Options) + ]); // Map the AG-UI agent endpoint app.MapAGUI("/", agent); diff --git a/dotnet/samples/AGUIClientServer/AGUIServer/Properties/launchSettings.json b/dotnet/samples/AGUIClientServer/AGUIServer/Properties/launchSettings.json new file mode 100644 index 0000000000..6e38bd9975 --- /dev/null +++ b/dotnet/samples/AGUIClientServer/AGUIServer/Properties/launchSettings.json @@ -0,0 +1,12 @@ +{ + "profiles": { + "AGUIServer": { + "commandName": "Project", + "launchBrowser": true, + "environmentVariables": { + "ASPNETCORE_ENVIRONMENT": "Development" + }, + "applicationUrl": "http://localhost:5100;https://localhost:5101" + } + } +} \ No newline at end of file diff --git a/dotnet/samples/AGUIClientServer/AGUIServer/ServerWeatherForecastRequest.cs b/dotnet/samples/AGUIClientServer/AGUIServer/ServerWeatherForecastRequest.cs new file mode 100644 index 0000000000..a4e3d983ca --- /dev/null +++ b/dotnet/samples/AGUIClientServer/AGUIServer/ServerWeatherForecastRequest.cs @@ -0,0 +1,9 @@ +// Copyright (c) Microsoft. All rights reserved. + +namespace AGUIServer; + +internal sealed class ServerWeatherForecastRequest +{ + public DateTime Date { get; set; } + public string Location { get; set; } = "Seattle"; +} diff --git a/dotnet/samples/AGUIClientServer/AGUIServer/ServerWeatherForecastResponse.cs b/dotnet/samples/AGUIClientServer/AGUIServer/ServerWeatherForecastResponse.cs new file mode 100644 index 0000000000..2bc5d8fbb9 --- /dev/null +++ b/dotnet/samples/AGUIClientServer/AGUIServer/ServerWeatherForecastResponse.cs @@ -0,0 +1,12 @@ +// Copyright (c) Microsoft. All rights reserved. + +namespace AGUIServer; + +internal sealed class ServerWeatherForecastResponse +{ + public string Summary { get; set; } = ""; + + public int TemperatureC { get; set; } + + public DateTime Date { get; set; } +} diff --git a/dotnet/samples/AGUIClientServer/README.md b/dotnet/samples/AGUIClientServer/README.md index dabc841542..b0ad2265d0 100644 --- a/dotnet/samples/AGUIClientServer/README.md +++ b/dotnet/samples/AGUIClientServer/README.md @@ -134,15 +134,21 @@ This automatically handles: ### Client Side -The `AGUIClient` uses the `AGUIAgent` class to connect to the remote server: +The `AGUIClient` uses the `AGUIChatClient` to connect to the remote server: ```csharp -AGUIAgent agent = new( - id: "agui-client", +using HttpClient httpClient = new(); +var chatClient = new AGUIChatClient( + httpClient, + endpoint: serverUrl, + modelId: "agui-client", + jsonSerializerOptions: null); + +AIAgent agent = chatClient.CreateAIAgent( + instructions: null, + name: "agui-client", description: "AG-UI Client Agent", - messages: [], - httpClient: httpClient, - endpoint: serverUrl); + tools: []); bool isFirstUpdate = true; AgentRunResponseUpdate? currentUpdate = null; diff --git a/dotnet/samples/AGUIWebChat/Client/AGUIWebChatClient.csproj b/dotnet/samples/AGUIWebChat/Client/AGUIWebChatClient.csproj new file mode 100644 index 0000000000..b28e53df6e --- /dev/null +++ b/dotnet/samples/AGUIWebChat/Client/AGUIWebChatClient.csproj @@ -0,0 +1,14 @@ + + + + net10.0 + enable + enable + true + + + + + + + diff --git a/dotnet/samples/AGUIWebChat/Client/Components/App.razor b/dotnet/samples/AGUIWebChat/Client/Components/App.razor new file mode 100644 index 0000000000..a64d576883 --- /dev/null +++ b/dotnet/samples/AGUIWebChat/Client/Components/App.razor @@ -0,0 +1,23 @@ + + + + + + + + + + + + + + + + + + + + +@code { + private readonly IComponentRenderMode renderMode = new InteractiveServerRenderMode(prerender: false); +} diff --git a/dotnet/samples/AGUIWebChat/Client/Components/Layout/LoadingSpinner.razor b/dotnet/samples/AGUIWebChat/Client/Components/Layout/LoadingSpinner.razor new file mode 100644 index 0000000000..116455ce45 --- /dev/null +++ b/dotnet/samples/AGUIWebChat/Client/Components/Layout/LoadingSpinner.razor @@ -0,0 +1 @@ +
diff --git a/dotnet/samples/AGUIWebChat/Client/Components/Layout/LoadingSpinner.razor.css b/dotnet/samples/AGUIWebChat/Client/Components/Layout/LoadingSpinner.razor.css new file mode 100644 index 0000000000..e599d27e86 --- /dev/null +++ b/dotnet/samples/AGUIWebChat/Client/Components/Layout/LoadingSpinner.razor.css @@ -0,0 +1,89 @@ +/* Used under CC0 license */ + +.lds-ellipsis { + color: #666; + animation: fade-in 1s; +} + +@keyframes fade-in { + 0% { + opacity: 0; + } + + 100% { + opacity: 1; + } +} + + .lds-ellipsis, + .lds-ellipsis div { + box-sizing: border-box; + } + +.lds-ellipsis { + margin: auto; + display: block; + position: relative; + width: 80px; + height: 80px; +} + + .lds-ellipsis div { + position: absolute; + top: 33.33333px; + width: 10px; + height: 10px; + border-radius: 50%; + background: currentColor; + animation-timing-function: cubic-bezier(0, 1, 1, 0); + } + + .lds-ellipsis div:nth-child(1) { + left: 8px; + animation: lds-ellipsis1 0.6s infinite; + } + + .lds-ellipsis div:nth-child(2) { + left: 8px; + animation: lds-ellipsis2 0.6s infinite; + } + + .lds-ellipsis div:nth-child(3) { + left: 32px; + animation: lds-ellipsis2 0.6s infinite; + } + + .lds-ellipsis div:nth-child(4) { + left: 56px; + animation: lds-ellipsis3 0.6s infinite; + } + +@keyframes lds-ellipsis1 { + 0% { + transform: scale(0); + } + + 100% { + transform: scale(1); + } +} + +@keyframes lds-ellipsis3 { + 0% { + transform: scale(1); + } + + 100% { + transform: scale(0); + } +} + +@keyframes lds-ellipsis2 { + 0% { + transform: translate(0, 0); + } + + 100% { + transform: translate(24px, 0); + } +} diff --git a/dotnet/samples/AGUIWebChat/Client/Components/Layout/MainLayout.razor b/dotnet/samples/AGUIWebChat/Client/Components/Layout/MainLayout.razor new file mode 100644 index 0000000000..f3da3cbae5 --- /dev/null +++ b/dotnet/samples/AGUIWebChat/Client/Components/Layout/MainLayout.razor @@ -0,0 +1,9 @@ +@inherits LayoutComponentBase + +@Body + +
+ An unhandled error has occurred. + Reload + 🗙 +
diff --git a/dotnet/samples/AGUIWebChat/Client/Components/Layout/MainLayout.razor.css b/dotnet/samples/AGUIWebChat/Client/Components/Layout/MainLayout.razor.css new file mode 100644 index 0000000000..60cec92d5e --- /dev/null +++ b/dotnet/samples/AGUIWebChat/Client/Components/Layout/MainLayout.razor.css @@ -0,0 +1,20 @@ +#blazor-error-ui { + color-scheme: light only; + background: lightyellow; + bottom: 0; + box-shadow: 0 -1px 2px rgba(0, 0, 0, 0.2); + box-sizing: border-box; + display: none; + left: 0; + padding: 0.6rem 1.25rem 0.7rem 1.25rem; + position: fixed; + width: 100%; + z-index: 1000; +} + + #blazor-error-ui .dismiss { + cursor: pointer; + position: absolute; + right: 0.75rem; + top: 0.5rem; + } diff --git a/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/Chat.razor b/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/Chat.razor new file mode 100644 index 0000000000..31eb7e406c --- /dev/null +++ b/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/Chat.razor @@ -0,0 +1,94 @@ +@page "/" +@using System.ComponentModel +@inject IChatClient ChatClient +@inject NavigationManager Nav +@implements IDisposable + +Chat + + + + + +
Ask the assistant a question to start a conversation.
+
+
+
+ + +
+ +@code { + private const string SystemPrompt = @" + You are a helpful assistant. + "; + + private int statefulMessageCount; + private readonly ChatOptions chatOptions = new(); + private readonly List messages = new(); + private CancellationTokenSource? currentResponseCancellation; + private ChatMessage? currentResponseMessage; + private ChatInput? chatInput; + private ChatSuggestions? chatSuggestions; + + protected override void OnInitialized() + { + statefulMessageCount = 0; + messages.Add(new(ChatRole.System, SystemPrompt)); + } + + private async Task AddUserMessageAsync(ChatMessage userMessage) + { + CancelAnyCurrentResponse(); + + // Add the user message to the conversation + messages.Add(userMessage); + chatSuggestions?.Clear(); + await chatInput!.FocusAsync(); + + // Stream and display a new response from the IChatClient + var responseText = new TextContent(""); + currentResponseMessage = new ChatMessage(ChatRole.Assistant, [responseText]); + StateHasChanged(); + currentResponseCancellation = new(); + await foreach (var update in ChatClient.GetStreamingResponseAsync(messages.Skip(statefulMessageCount), chatOptions, currentResponseCancellation.Token)) + { + messages.AddMessages(update, filter: c => c is not TextContent); + responseText.Text += update.Text; + chatOptions.ConversationId = update.ConversationId; + ChatMessageItem.NotifyChanged(currentResponseMessage); + } + + // Store the final response in the conversation, and begin getting suggestions + messages.Add(currentResponseMessage!); + statefulMessageCount = chatOptions.ConversationId is not null ? messages.Count : 0; + currentResponseMessage = null; + chatSuggestions?.Update(messages); + } + + private void CancelAnyCurrentResponse() + { + // If a response was cancelled while streaming, include it in the conversation so it's not lost + if (currentResponseMessage is not null) + { + messages.Add(currentResponseMessage); + } + + currentResponseCancellation?.Cancel(); + currentResponseMessage = null; + } + + private async Task ResetConversationAsync() + { + CancelAnyCurrentResponse(); + messages.Clear(); + messages.Add(new(ChatRole.System, SystemPrompt)); + chatOptions.ConversationId = null; + statefulMessageCount = 0; + chatSuggestions?.Clear(); + await chatInput!.FocusAsync(); + } + + public void Dispose() + => currentResponseCancellation?.Cancel(); +} diff --git a/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/Chat.razor.css b/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/Chat.razor.css new file mode 100644 index 0000000000..08841605f6 --- /dev/null +++ b/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/Chat.razor.css @@ -0,0 +1,11 @@ +.chat-container { + position: sticky; + bottom: 0; + padding-left: 1.5rem; + padding-right: 1.5rem; + padding-top: 0.75rem; + padding-bottom: 1.5rem; + border-top-width: 1px; + background-color: #F3F4F6; + border-color: #E5E7EB; +} diff --git a/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/ChatCitation.razor b/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/ChatCitation.razor new file mode 100644 index 0000000000..ccb5853cec --- /dev/null +++ b/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/ChatCitation.razor @@ -0,0 +1,38 @@ +@using System.Web +@if (!string.IsNullOrWhiteSpace(viewerUrl)) +{ + + + + +
+
@File
+
@Quote
+
+
+} + +@code { + [Parameter] + public required string File { get; set; } + + [Parameter] + public int? PageNumber { get; set; } + + [Parameter] + public required string Quote { get; set; } + + private string? viewerUrl; + + protected override void OnParametersSet() + { + viewerUrl = null; + + // If you ingest other types of content besides PDF files, construct a URL to an appropriate viewer here + if (File.EndsWith(".pdf")) + { + var search = Quote?.Trim('.', ',', ' ', '\n', '\r', '\t', '"', '\''); + viewerUrl = $"lib/pdf_viewer/viewer.html?file=/Data/{HttpUtility.UrlEncode(File)}#page={PageNumber}&search={HttpUtility.UrlEncode(search)}&phrase=true"; + } + } +} diff --git a/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/ChatCitation.razor.css b/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/ChatCitation.razor.css new file mode 100644 index 0000000000..763c82aec4 --- /dev/null +++ b/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/ChatCitation.razor.css @@ -0,0 +1,37 @@ +.citation { + display: inline-flex; + padding-top: 0.5rem; + padding-bottom: 0.5rem; + padding-left: 0.75rem; + padding-right: 0.75rem; + margin-top: 1rem; + margin-right: 1rem; + border-bottom: 2px solid #a770de; + gap: 0.5rem; + border-radius: 0.25rem; + font-size: 0.875rem; + line-height: 1.25rem; + background-color: #ffffff; +} + + .citation[href]:hover { + outline: 1px solid #865cb1; + } + + .citation svg { + width: 1.5rem; + height: 1.5rem; + } + + .citation:active { + background-color: rgba(0,0,0,0.05); + } + +.citation-content { + display: flex; + flex-direction: column; +} + +.citation-file { + font-weight: 600; +} diff --git a/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/ChatHeader.razor b/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/ChatHeader.razor new file mode 100644 index 0000000000..a339038e2a --- /dev/null +++ b/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/ChatHeader.razor @@ -0,0 +1,17 @@ +
+
+ +
+ +

AGUI WebChat

+
+ +@code { + [Parameter] + public EventCallback OnNewChat { get; set; } +} diff --git a/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/ChatHeader.razor.css b/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/ChatHeader.razor.css new file mode 100644 index 0000000000..97f0a8d43a --- /dev/null +++ b/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/ChatHeader.razor.css @@ -0,0 +1,25 @@ +.chat-header-container { + top: 0; + padding: 1.5rem; +} + +.chat-header-controls { + margin-bottom: 1.5rem; +} + +h1 { + overflow: hidden; + text-overflow: ellipsis; +} + +.new-chat-icon { + width: 1.25rem; + height: 1.25rem; + color: rgb(55, 65, 81); +} + +@media (min-width: 768px) { + .chat-header-container { + position: sticky; + } +} diff --git a/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/ChatInput.razor b/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/ChatInput.razor new file mode 100644 index 0000000000..e87ac6ccf4 --- /dev/null +++ b/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/ChatInput.razor @@ -0,0 +1,51 @@ +@inject IJSRuntime JS + + + + + +@code { + private ElementReference textArea; + private string? messageText; + + [Parameter] + public EventCallback OnSend { get; set; } + + public ValueTask FocusAsync() + => textArea.FocusAsync(); + + private async Task SendMessageAsync() + { + if (messageText is { Length: > 0 } text) + { + messageText = null; + await OnSend.InvokeAsync(new ChatMessage(ChatRole.User, text)); + } + } + + protected override async Task OnAfterRenderAsync(bool firstRender) + { + if (firstRender) + { + try + { + var module = await JS.InvokeAsync("import", "./Components/Pages/Chat/ChatInput.razor.js"); + await module.InvokeVoidAsync("init", textArea); + await module.DisposeAsync(); + } + catch (JSDisconnectedException) + { + } + } + } +} diff --git a/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/ChatInput.razor.css b/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/ChatInput.razor.css new file mode 100644 index 0000000000..375dd711d9 --- /dev/null +++ b/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/ChatInput.razor.css @@ -0,0 +1,57 @@ +.input-box { + display: flex; + flex-direction: column; + background: white; + border: 1px solid rgb(229, 231, 235); + border-radius: 8px; + padding: 0.5rem 0.75rem; + margin-top: 0.75rem; +} + + .input-box:focus-within { + outline: 2px solid #4152d5; + } + +textarea { + resize: none; + border: none; + outline: none; + flex-grow: 1; +} + + textarea:placeholder-shown + .tools { + --send-button-color: #aaa; + } + +.tools { + display: flex; + margin-top: 1rem; + align-items: center; +} + +.tool-icon { + width: 1.25rem; + height: 1.25rem; +} + +.send-button { + color: var(--send-button-color); + margin-left: auto; +} + + .send-button:hover { + color: black; + } + +.attach { + background-color: white; + border-style: dashed; + color: #888; + border-color: #888; + padding: 3px 8px; +} + + .attach:hover { + background-color: #f0f0f0; + color: black; + } diff --git a/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/ChatInput.razor.js b/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/ChatInput.razor.js new file mode 100644 index 0000000000..e4bd8af20a --- /dev/null +++ b/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/ChatInput.razor.js @@ -0,0 +1,43 @@ +export function init(elem) { + elem.focus(); + + // Auto-resize whenever the user types or if the value is set programmatically + elem.addEventListener('input', () => resizeToFit(elem)); + afterPropertyWritten(elem, 'value', () => resizeToFit(elem)); + + // Auto-submit the form on 'enter' keypress + elem.addEventListener('keydown', (e) => { + if (e.key === 'Enter' && !e.shiftKey) { + e.preventDefault(); + elem.dispatchEvent(new CustomEvent('change', { bubbles: true })); + elem.closest('form').dispatchEvent(new CustomEvent('submit', { bubbles: true, cancelable: true })); + } + }); +} + +function resizeToFit(elem) { + const lineHeight = parseFloat(getComputedStyle(elem).lineHeight); + + elem.rows = 1; + const numLines = Math.ceil(elem.scrollHeight / lineHeight); + elem.rows = Math.min(5, Math.max(1, numLines)); +} + +function afterPropertyWritten(target, propName, callback) { + const descriptor = getPropertyDescriptor(target, propName); + Object.defineProperty(target, propName, { + get: function () { + return descriptor.get.apply(this, arguments); + }, + set: function () { + const result = descriptor.set.apply(this, arguments); + callback(); + return result; + } + }); +} + +function getPropertyDescriptor(target, propertyName) { + return Object.getOwnPropertyDescriptor(target, propertyName) + || getPropertyDescriptor(Object.getPrototypeOf(target), propertyName); +} diff --git a/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/ChatMessageItem.razor b/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/ChatMessageItem.razor new file mode 100644 index 0000000000..6f4e1357c9 --- /dev/null +++ b/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/ChatMessageItem.razor @@ -0,0 +1,73 @@ +@using System.Runtime.CompilerServices +@using System.Text.RegularExpressions +@using System.Linq + +@if (Message.Role == ChatRole.User) +{ +
+ @Message.Text +
+} +else if (Message.Role == ChatRole.Assistant) +{ + foreach (var content in Message.Contents) + { + if (content is TextContent { Text: { Length: > 0 } text }) + { +
+
+
+ + + +
+
+
Assistant
+
+
@((MarkupString)text)
+
+
+ } + else if (content is FunctionCallContent { Name: "Search" } fcc && fcc.Arguments?.TryGetValue("searchPhrase", out var searchPhrase) is true) + { + + } + } +} + +@code { + private static readonly ConditionalWeakTable SubscribersLookup = new(); + + [Parameter, EditorRequired] + public required ChatMessage Message { get; set; } + + [Parameter] + public bool InProgress { get; set;} + + protected override void OnInitialized() + { + SubscribersLookup.AddOrUpdate(Message, this); + } + + public static void NotifyChanged(ChatMessage source) + { + if (SubscribersLookup.TryGetValue(source, out var subscriber)) + { + subscriber.StateHasChanged(); + } + } +} diff --git a/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/ChatMessageItem.razor.css b/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/ChatMessageItem.razor.css new file mode 100644 index 0000000000..16443cf657 --- /dev/null +++ b/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/ChatMessageItem.razor.css @@ -0,0 +1,67 @@ +.user-message { + background: rgb(182 215 232); + align-self: flex-end; + min-width: 25%; + max-width: calc(100% - 5rem); + padding: 0.5rem 1.25rem; + border-radius: 0.25rem; + color: #1F2937; + white-space: pre-wrap; +} + +.assistant-message, .assistant-search { + display: grid; + grid-template-rows: min-content; + grid-template-columns: 2rem minmax(0, 1fr); + gap: 0.25rem; +} + +.assistant-message-header { + font-weight: 600; +} + +.assistant-message-text { + grid-column-start: 2; +} + +.assistant-message-icon { + display: flex; + justify-content: center; + align-items: center; + border-radius: 9999px; + width: 1.5rem; + height: 1.5rem; + color: #ffffff; + background: #9b72ce; +} + + .assistant-message-icon svg { + width: 1rem; + height: 1rem; + } + +.assistant-search { + font-size: 0.875rem; + line-height: 1.25rem; +} + +.assistant-search-icon { + display: flex; + justify-content: center; + align-items: center; + width: 1.5rem; + height: 1.5rem; +} + + .assistant-search-icon svg { + width: 1rem; + height: 1rem; + } + +.assistant-search-content { + align-content: center; +} + +.assistant-search-phrase { + font-weight: 600; +} diff --git a/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/ChatMessageList.razor b/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/ChatMessageList.razor new file mode 100644 index 0000000000..d245f455f1 --- /dev/null +++ b/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/ChatMessageList.razor @@ -0,0 +1,42 @@ +@inject IJSRuntime JS + +
+ + @foreach (var message in Messages) + { + + } + + @if (InProgressMessage is not null) + { + + + } + else if (IsEmpty) + { +
@NoMessagesContent
+ } +
+
+ +@code { + [Parameter] + public required IEnumerable Messages { get; set; } + + [Parameter] + public ChatMessage? InProgressMessage { get; set; } + + [Parameter] + public RenderFragment? NoMessagesContent { get; set; } + + private bool IsEmpty => !Messages.Any(m => (m.Role == ChatRole.User || m.Role == ChatRole.Assistant) && !string.IsNullOrEmpty(m.Text)); + + protected override async Task OnAfterRenderAsync(bool firstRender) + { + if (firstRender) + { + // Activates the auto-scrolling behavior + await JS.InvokeVoidAsync("import", "./Components/Pages/Chat/ChatMessageList.razor.js"); + } + } +} diff --git a/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/ChatMessageList.razor.css b/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/ChatMessageList.razor.css new file mode 100644 index 0000000000..4be50ddfc3 --- /dev/null +++ b/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/ChatMessageList.razor.css @@ -0,0 +1,22 @@ +.message-list-container { + margin: 2rem 1.5rem; + flex-grow: 1; +} + +.message-list { + display: flex; + flex-direction: column; + gap: 1.25rem; +} + +.no-messages { + text-align: center; + font-size: 1.25rem; + color: #999; + margin-top: calc(40vh - 18rem); +} + +chat-messages > ::deep div:last-of-type { + /* Adds some vertical buffer to so that suggestions don't overlap the output when they appear */ + margin-bottom: 2rem; +} diff --git a/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/ChatMessageList.razor.js b/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/ChatMessageList.razor.js new file mode 100644 index 0000000000..9755d47c29 --- /dev/null +++ b/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/ChatMessageList.razor.js @@ -0,0 +1,34 @@ +// The following logic provides auto-scroll behavior for the chat messages list. +// If you don't want that behavior, you can simply not load this module. + +window.customElements.define('chat-messages', class ChatMessages extends HTMLElement { + static _isFirstAutoScroll = true; + + connectedCallback() { + this._observer = new MutationObserver(mutations => this._scheduleAutoScroll(mutations)); + this._observer.observe(this, { childList: true, attributes: true }); + } + + disconnectedCallback() { + this._observer.disconnect(); + } + + _scheduleAutoScroll(mutations) { + // Debounce the calls in case multiple DOM updates occur together + cancelAnimationFrame(this._nextAutoScroll); + this._nextAutoScroll = requestAnimationFrame(() => { + const addedUserMessage = mutations.some(m => Array.from(m.addedNodes).some(n => n.parentElement === this && n.classList?.contains('user-message'))); + const elem = this.lastElementChild; + if (ChatMessages._isFirstAutoScroll || addedUserMessage || this._elemIsNearScrollBoundary(elem, 300)) { + elem.scrollIntoView({ behavior: ChatMessages._isFirstAutoScroll ? 'instant' : 'smooth' }); + ChatMessages._isFirstAutoScroll = false; + } + }); + } + + _elemIsNearScrollBoundary(elem, threshold) { + const maxScrollPos = document.body.scrollHeight - window.innerHeight; + const remainingScrollDistance = maxScrollPos - window.scrollY; + return remainingScrollDistance < elem.offsetHeight + threshold; + } +}); diff --git a/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/ChatSuggestions.razor b/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/ChatSuggestions.razor new file mode 100644 index 0000000000..69ca922a8c --- /dev/null +++ b/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/ChatSuggestions.razor @@ -0,0 +1,78 @@ +@inject IChatClient ChatClient + +@if (suggestions is not null) +{ +
+ @foreach (var suggestion in suggestions) + { + + } +
+} + +@code { + private static string Prompt = @" + Suggest up to 3 follow-up questions that I could ask you to help me complete my task. + Each suggestion must be a complete sentence, maximum 6 words. + Each suggestion must be phrased as something that I (the user) would ask you (the assistant) in response to your previous message, + for example 'How do I do that?' or 'Explain ...'. + If there are no suggestions, reply with an empty list. + "; + + private string[]? suggestions; + private CancellationTokenSource? cancellation; + + [Parameter] + public EventCallback OnSelected { get; set; } + + public void Clear() + { + suggestions = null; + cancellation?.Cancel(); + } + + public void Update(IReadOnlyList messages) + { + // Runs in the background and handles its own cancellation/errors + _ = UpdateSuggestionsAsync(messages); + } + + private async Task UpdateSuggestionsAsync(IReadOnlyList messages) + { + cancellation?.Cancel(); + cancellation = new CancellationTokenSource(); + + try + { + var response = await ChatClient.GetResponseAsync( + [.. ReduceMessages(messages), new(ChatRole.User, Prompt)], + cancellationToken: cancellation.Token); + if (!response.TryGetResult(out suggestions)) + { + suggestions = null; + } + + StateHasChanged(); + } + catch (Exception ex) when (ex is not OperationCanceledException) + { + await DispatchExceptionAsync(ex); + } + } + + private async Task AddSuggestionAsync(string text) + { + await OnSelected.InvokeAsync(new(ChatRole.User, text)); + } + + private IEnumerable ReduceMessages(IReadOnlyList messages) + { + // Get any leading system messages, plus up to 5 user/assistant messages + // This should be enough context to generate suggestions without unnecessarily resending entire conversations when long + var systemMessages = messages.TakeWhile(m => m.Role == ChatRole.System); + var otherMessages = messages.Where((m, index) => m.Role == ChatRole.User || m.Role == ChatRole.Assistant).Where(m => !string.IsNullOrEmpty(m.Text)).TakeLast(5); + return systemMessages.Concat(otherMessages); + } +} diff --git a/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/ChatSuggestions.razor.css b/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/ChatSuggestions.razor.css new file mode 100644 index 0000000000..dcc7ee8bd8 --- /dev/null +++ b/dotnet/samples/AGUIWebChat/Client/Components/Pages/Chat/ChatSuggestions.razor.css @@ -0,0 +1,9 @@ +.suggestions { + text-align: right; + white-space: nowrap; + gap: 0.5rem; + justify-content: flex-end; + flex-wrap: wrap; + display: flex; + margin-bottom: 0.75rem; +} diff --git a/dotnet/samples/AGUIWebChat/Client/Components/Routes.razor b/dotnet/samples/AGUIWebChat/Client/Components/Routes.razor new file mode 100644 index 0000000000..faa2a8c2d5 --- /dev/null +++ b/dotnet/samples/AGUIWebChat/Client/Components/Routes.razor @@ -0,0 +1,6 @@ + + + + + + diff --git a/dotnet/samples/AGUIWebChat/Client/Components/_Imports.razor b/dotnet/samples/AGUIWebChat/Client/Components/_Imports.razor new file mode 100644 index 0000000000..82be3d448e --- /dev/null +++ b/dotnet/samples/AGUIWebChat/Client/Components/_Imports.razor @@ -0,0 +1,12 @@ +@using System.Net.Http +@using System.Net.Http.Json +@using Microsoft.AspNetCore.Components.Forms +@using Microsoft.AspNetCore.Components.Routing +@using Microsoft.AspNetCore.Components.Web +@using static Microsoft.AspNetCore.Components.Web.RenderMode +@using Microsoft.AspNetCore.Components.Web.Virtualization +@using Microsoft.JSInterop +@using AGUIWebChatClient +@using AGUIWebChatClient.Components +@using AGUIWebChatClient.Components.Layout +@using Microsoft.Extensions.AI diff --git a/dotnet/samples/AGUIWebChat/Client/Program.cs b/dotnet/samples/AGUIWebChat/Client/Program.cs new file mode 100644 index 0000000000..c145227062 --- /dev/null +++ b/dotnet/samples/AGUIWebChat/Client/Program.cs @@ -0,0 +1,34 @@ +// Copyright (c) Microsoft. All rights reserved. + +using AGUIWebChatClient.Components; +using Microsoft.Agents.AI.AGUI; + +WebApplicationBuilder builder = WebApplication.CreateBuilder(args); + +// Add services to the container. +builder.Services.AddRazorComponents() + .AddInteractiveServerComponents(); + +string serverUrl = builder.Configuration["SERVER_URL"] ?? "http://localhost:5100"; + +builder.Services.AddHttpClient("aguiserver", httpClient => httpClient.BaseAddress = new Uri(serverUrl)); + +builder.Services.AddChatClient(sp => new AGUIChatClient( + sp.GetRequiredService().CreateClient("aguiserver"), "ag-ui")); + +WebApplication app = builder.Build(); + +// Configure the HTTP request pipeline. +if (!app.Environment.IsDevelopment()) +{ + app.UseExceptionHandler("/Error", createScopeForErrors: true); + app.UseHsts(); +} + +app.UseHttpsRedirection(); +app.UseAntiforgery(); +app.MapStaticAssets(); +app.MapRazorComponents() + .AddInteractiveServerRenderMode(); + +app.Run(); diff --git a/dotnet/samples/AGUIWebChat/Client/Properties/launchSettings.json b/dotnet/samples/AGUIWebChat/Client/Properties/launchSettings.json new file mode 100644 index 0000000000..348e16bc3b --- /dev/null +++ b/dotnet/samples/AGUIWebChat/Client/Properties/launchSettings.json @@ -0,0 +1,15 @@ +{ + "$schema": "https://json.schemastore.org/launchsettings.json", + "profiles": { + "http": { + "commandName": "Project", + "dotnetRunMessages": true, + "launchBrowser": true, + "applicationUrl": "http://localhost:5000", + "environmentVariables": { + "ASPNETCORE_ENVIRONMENT": "Development", + "SERVER_URL": "http://localhost:5100" + } + } + } +} diff --git a/dotnet/samples/AGUIWebChat/Client/wwwroot/app.css b/dotnet/samples/AGUIWebChat/Client/wwwroot/app.css new file mode 100644 index 0000000000..5fd82f3bb0 --- /dev/null +++ b/dotnet/samples/AGUIWebChat/Client/wwwroot/app.css @@ -0,0 +1,93 @@ +html { + min-height: 100vh; +} + +html, .main-background-gradient { + background: linear-gradient(to bottom, rgb(225 227 233), #f4f4f4 25rem); +} + +body { + display: flex; + flex-direction: column; + min-height: 100vh; + font-family: 'Segoe UI', Tahoma, Geneva, Verdana, sans-serif; +} + +html::after { + content: ''; + background-image: linear-gradient(to right, #3a4ed5, #3acfd5 15%, #d53abf 85%, red); + width: 100%; + height: 2px; + position: fixed; + top: 0; +} + +h1 { + font-size: 2.25rem; + line-height: 2.5rem; + font-weight: 600; +} + +h1:focus { + outline: none; +} + +.valid.modified:not([type=checkbox]) { + outline: 1px solid #26b050; +} + +.invalid { + outline: 1px solid #e50000; +} + +.validation-message { + color: #e50000; +} + +.blazor-error-boundary { + background: url() no-repeat 1rem/1.8rem, #b32121; + padding: 1rem 1rem 1rem 3.7rem; + color: white; +} + + .blazor-error-boundary::after { + content: "An error has occurred." + } + +.btn-default { + display: flex; + padding: 0.25rem 0.75rem; + gap: 0.25rem; + align-items: center; + border-radius: 0.25rem; + border: 1px solid #9CA3AF; + font-size: 0.875rem; + line-height: 1.25rem; + font-weight: 600; + background-color: #D1D5DB; +} + + .btn-default:hover { + background-color: #E5E7EB; + } + +.btn-subtle { + display: flex; + padding: 0.25rem 0.75rem; + gap: 0.25rem; + align-items: center; + border-radius: 0.25rem; + border: 1px solid #D1D5DB; + font-size: 0.875rem; + line-height: 1.25rem; +} + + .btn-subtle:hover { + border-color: #93C5FD; + background-color: #DBEAFE; + } + +.page-width { + max-width: 1024px; + margin: auto; +} diff --git a/dotnet/samples/AGUIWebChat/Client/wwwroot/favicon.png b/dotnet/samples/AGUIWebChat/Client/wwwroot/favicon.png new file mode 100644 index 0000000000..8422b59695 Binary files /dev/null and b/dotnet/samples/AGUIWebChat/Client/wwwroot/favicon.png differ diff --git a/dotnet/samples/AGUIWebChat/README.md b/dotnet/samples/AGUIWebChat/README.md new file mode 100644 index 0000000000..75af0872c1 --- /dev/null +++ b/dotnet/samples/AGUIWebChat/README.md @@ -0,0 +1,185 @@ +# AGUI WebChat Sample + +This sample demonstrates a Blazor-based web chat application using the AG-UI protocol to communicate with an AI agent server. + +The sample consists of two projects: + +1. **Server** - An ASP.NET Core server that hosts a simple chat agent using the AG-UI protocol +2. **Client** - A Blazor Server application with a rich chat UI for interacting with the agent + +## Prerequisites + +### Azure OpenAI Configuration + +The server requires Azure OpenAI credentials. Set the following environment variables: + +```powershell +$env:AZURE_OPENAI_ENDPOINT="https://your-resource.openai.azure.com/" +$env:AZURE_OPENAI_DEPLOYMENT_NAME="your-deployment-name" # e.g., "gpt-4o" +``` + +The server uses `DefaultAzureCredential` for authentication. Ensure you are logged in using one of the following methods: + +- Azure CLI: `az login` +- Azure PowerShell: `Connect-AzAccount` +- Visual Studio or VS Code with Azure extensions +- Environment variables with service principal credentials + +## Running the Sample + +### Step 1: Start the Server + +Open a terminal and navigate to the Server directory: + +```powershell +cd Server +dotnet run +``` + +The server will start on `http://localhost:5100` and expose the AG-UI endpoint at `/ag-ui`. + +### Step 2: Start the Client + +Open a new terminal and navigate to the Client directory: + +```powershell +cd Client +dotnet run +``` + +The client will start on `http://localhost:5000`. Open your browser and navigate to `http://localhost:5000` to access the chat interface. + +### Step 3: Chat with the Agent + +Type your message in the text box at the bottom of the page and press Enter or click the send button. The assistant will respond with streaming text that appears in real-time. + +Features: +- **Streaming responses**: Watch the assistant's response appear word by word +- **Conversation suggestions**: The assistant may offer follow-up questions after responding +- **New chat**: Click the "New chat" button to start a fresh conversation +- **Auto-scrolling**: The chat automatically scrolls to show new messages + +## How It Works + +### Server (AG-UI Host) + +The server (`Server/Program.cs`) creates a simple chat agent: + +```csharp +// Create Azure OpenAI client +AzureOpenAIClient azureOpenAIClient = new AzureOpenAIClient( + new Uri(endpoint), + new DefaultAzureCredential()); + +ChatClient chatClient = azureOpenAIClient.GetChatClient(deploymentName); + +// Create AI agent +ChatClientAgent agent = chatClient.AsIChatClient().CreateAIAgent( + name: "ChatAssistant", + instructions: "You are a helpful assistant."); + +// Map AG-UI endpoint +app.MapAGUI("/ag-ui", agent); +``` + +The server exposes the agent via the AG-UI protocol at `http://localhost:5100/ag-ui`. + +### Client (Blazor Web App) + +The client (`Client/Program.cs`) configures an `AGUIChatClient` to connect to the server: + +```csharp +string serverUrl = builder.Configuration["SERVER_URL"] ?? "http://localhost:5100"; + +builder.Services.AddHttpClient("aguiserver", httpClient => httpClient.BaseAddress = new Uri(serverUrl)); + +builder.Services.AddChatClient(sp => new AGUIChatClient( + sp.GetRequiredService().CreateClient("aguiserver"), "ag-ui")); +``` + +The Blazor UI (`Client/Components/Pages/Chat/Chat.razor`) uses the `IChatClient` to: +- Send user messages to the agent +- Stream responses back in real-time +- Maintain conversation history +- Display messages with appropriate styling + +### UI Components + +The chat interface is built from several Blazor components: + +- **Chat.razor** - Main chat page coordinating the conversation flow +- **ChatHeader.razor** - Header with "New chat" button +- **ChatMessageList.razor** - Scrollable list of messages with auto-scroll +- **ChatMessageItem.razor** - Individual message rendering (user vs assistant) +- **ChatInput.razor** - Text input with auto-resize and keyboard shortcuts +- **ChatSuggestions.razor** - AI-generated follow-up question suggestions +- **LoadingSpinner.razor** - Animated loading indicator during streaming + +## Configuration + +### Server Configuration + +The server URL and port are configured in `Server/Properties/launchSettings.json`: + +```json +{ + "profiles": { + "http": { + "applicationUrl": "http://localhost:5100" + } + } +} +``` + +### Client Configuration + +The client connects to the server URL specified in `Client/Properties/launchSettings.json`: + +```json +{ + "profiles": { + "http": { + "applicationUrl": "http://localhost:5000", + "environmentVariables": { + "SERVER_URL": "http://localhost:5100" + } + } + } +} +``` + +To change the server URL, modify the `SERVER_URL` environment variable in the client's launch settings or provide it at runtime: + +```powershell +$env:SERVER_URL="http://your-server:5100" +dotnet run +``` + +## Customization + +### Changing the Agent Instructions + +Edit the instructions in `Server/Program.cs`: + +```csharp +ChatClientAgent agent = chatClient.AsIChatClient().CreateAIAgent( + name: "ChatAssistant", + instructions: "You are a helpful coding assistant specializing in C# and .NET."); +``` + +### Styling the UI + +The chat interface uses CSS files colocated with each Razor component. Key styles: + +- `wwwroot/app.css` - Global styles, buttons, color scheme +- `Components/Pages/Chat/Chat.razor.css` - Chat container layout +- `Components/Pages/Chat/ChatMessageItem.razor.css` - Message bubbles and icons +- `Components/Pages/Chat/ChatInput.razor.css` - Input box styling + +### Disabling Suggestions + +To disable the AI-generated follow-up suggestions, comment out the suggestions component in `Chat.razor`: + +```razor +@* *@ +``` diff --git a/dotnet/samples/Catalog/AgentsInWorkflows/AgentsInWorkflows.csproj b/dotnet/samples/AGUIWebChat/Server/AGUIWebChatServer.csproj similarity index 55% rename from dotnet/samples/Catalog/AgentsInWorkflows/AgentsInWorkflows.csproj rename to dotnet/samples/AGUIWebChat/Server/AGUIWebChatServer.csproj index f192c19901..c45adfd4a8 100644 --- a/dotnet/samples/Catalog/AgentsInWorkflows/AgentsInWorkflows.csproj +++ b/dotnet/samples/AGUIWebChat/Server/AGUIWebChatServer.csproj @@ -1,23 +1,21 @@ - + Exe - net9.0 - - enable + net10.0 enable + enable - - - - + + + diff --git a/dotnet/samples/AGUIWebChat/Server/Program.cs b/dotnet/samples/AGUIWebChat/Server/Program.cs new file mode 100644 index 0000000000..1683a7e3ed --- /dev/null +++ b/dotnet/samples/AGUIWebChat/Server/Program.cs @@ -0,0 +1,35 @@ +// Copyright (c) Microsoft. All rights reserved. + +// This sample demonstrates a basic AG-UI server hosting a chat agent for the Blazor web client. + +using Azure.AI.OpenAI; +using Azure.Identity; +using Microsoft.Agents.AI; +using Microsoft.Agents.AI.Hosting.AGUI.AspNetCore; +using Microsoft.Extensions.AI; +using OpenAI.Chat; + +WebApplicationBuilder builder = WebApplication.CreateBuilder(args); +builder.Services.AddHttpClient().AddLogging(); +builder.Services.AddAGUI(); + +WebApplication app = builder.Build(); + +string endpoint = builder.Configuration["AZURE_OPENAI_ENDPOINT"] ?? throw new InvalidOperationException("AZURE_OPENAI_ENDPOINT is not set."); +string deploymentName = builder.Configuration["AZURE_OPENAI_DEPLOYMENT_NAME"] ?? throw new InvalidOperationException("AZURE_OPENAI_DEPLOYMENT_NAME is not set."); + +// Create the AI agent +AzureOpenAIClient azureOpenAIClient = new( + new Uri(endpoint), + new DefaultAzureCredential()); + +ChatClient chatClient = azureOpenAIClient.GetChatClient(deploymentName); + +ChatClientAgent agent = chatClient.AsIChatClient().CreateAIAgent( + name: "ChatAssistant", + instructions: "You are a helpful assistant."); + +// Map the AG-UI agent endpoint +app.MapAGUI("/ag-ui", agent); + +await app.RunAsync(); diff --git a/dotnet/samples/AGUIWebChat/Server/Properties/launchSettings.json b/dotnet/samples/AGUIWebChat/Server/Properties/launchSettings.json new file mode 100644 index 0000000000..4d84174f7a --- /dev/null +++ b/dotnet/samples/AGUIWebChat/Server/Properties/launchSettings.json @@ -0,0 +1,14 @@ +{ + "$schema": "https://json.schemastore.org/launchsettings.json", + "profiles": { + "http": { + "commandName": "Project", + "dotnetRunMessages": true, + "launchBrowser": false, + "applicationUrl": "http://localhost:5100", + "environmentVariables": { + "ASPNETCORE_ENVIRONMENT": "Development" + } + } + } +} diff --git a/dotnet/samples/AgentWebChat/AgentWebChat.AgentHost/ActorFrameworkWebApplicationExtensions.cs b/dotnet/samples/AgentWebChat/AgentWebChat.AgentHost/ActorFrameworkWebApplicationExtensions.cs index 5e997c4f58..09e19a82f5 100644 --- a/dotnet/samples/AgentWebChat/AgentWebChat.AgentHost/ActorFrameworkWebApplicationExtensions.cs +++ b/dotnet/samples/AgentWebChat/AgentWebChat.AgentHost/ActorFrameworkWebApplicationExtensions.cs @@ -2,7 +2,7 @@ using System.Diagnostics.CodeAnalysis; using System.Text.Json.Serialization; -using Microsoft.Agents.AI.Hosting; +using Microsoft.Agents.AI; namespace AgentWebChat.AgentHost; @@ -10,24 +10,24 @@ internal static class ActorFrameworkWebApplicationExtensions { public static void MapAgentDiscovery(this IEndpointRouteBuilder endpoints, [StringSyntax("Route")] string path) { + var registeredAIAgents = endpoints.ServiceProvider.GetKeyedServices(KeyedService.AnyKey); + var routeGroup = endpoints.MapGroup(path); - routeGroup.MapGet("/", async ( - AgentCatalog agentCatalog, - CancellationToken cancellationToken) => + routeGroup.MapGet("/", async (CancellationToken cancellationToken) => + { + var results = new List(); + foreach (var result in registeredAIAgents) { - var results = new List(); - await foreach (var result in agentCatalog.GetAgentsAsync(cancellationToken).ConfigureAwait(false)) + results.Add(new AgentDiscoveryCard { - results.Add(new AgentDiscoveryCard - { - Name = result.Name!, - Description = result.Description, - }); - } + Name = result.Name!, + Description = result.Description, + }); + } - return Results.Ok(results); - }) - .WithName("GetAgents"); + return Results.Ok(results); + }) + .WithName("GetAgents"); } internal sealed class AgentDiscoveryCard diff --git a/dotnet/samples/AgentWebChat/AgentWebChat.AgentHost/AgentWebChat.AgentHost.csproj b/dotnet/samples/AgentWebChat/AgentWebChat.AgentHost/AgentWebChat.AgentHost.csproj index 802c864c1f..3f2a832a69 100644 --- a/dotnet/samples/AgentWebChat/AgentWebChat.AgentHost/AgentWebChat.AgentHost.csproj +++ b/dotnet/samples/AgentWebChat/AgentWebChat.AgentHost/AgentWebChat.AgentHost.csproj @@ -1,19 +1,20 @@ - + - net9.0 + net10.0 enable enable true + - + @@ -30,11 +31,4 @@
- - - - - - - - + \ No newline at end of file diff --git a/dotnet/samples/AgentWebChat/AgentWebChat.AgentHost/Custom/CustomAITools.cs b/dotnet/samples/AgentWebChat/AgentWebChat.AgentHost/Custom/CustomAITools.cs new file mode 100644 index 0000000000..14f0bcee41 --- /dev/null +++ b/dotnet/samples/AgentWebChat/AgentWebChat.AgentHost/Custom/CustomAITools.cs @@ -0,0 +1,15 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Microsoft.Extensions.AI; + +namespace AgentWebChat.AgentHost.Custom; + +public class CustomAITool : AITool; + +public class CustomFunctionTool : AIFunction +{ + protected override ValueTask InvokeCoreAsync(AIFunctionArguments arguments, CancellationToken cancellationToken) + { + return new ValueTask(arguments.Context?.Count ?? 0); + } +} diff --git a/dotnet/samples/AgentWebChat/AgentWebChat.AgentHost/Program.cs b/dotnet/samples/AgentWebChat/AgentWebChat.AgentHost/Program.cs index d86c53958d..7447c54aa1 100644 --- a/dotnet/samples/AgentWebChat/AgentWebChat.AgentHost/Program.cs +++ b/dotnet/samples/AgentWebChat/AgentWebChat.AgentHost/Program.cs @@ -2,8 +2,10 @@ using A2A.AspNetCore; using AgentWebChat.AgentHost; +using AgentWebChat.AgentHost.Custom; using AgentWebChat.AgentHost.Utilities; using Microsoft.Agents.AI; +using Microsoft.Agents.AI.DevUI; using Microsoft.Agents.AI.Hosting; using Microsoft.Agents.AI.Workflows; using Microsoft.Extensions.AI; @@ -20,11 +22,20 @@ // Configure the chat model and our agent. builder.AddKeyedChatClient("chat-model"); +// Add DevUI services +builder.AddDevUI(); + +// Add OpenAI services +builder.AddOpenAIChatCompletions(); +builder.AddOpenAIResponses(); + var pirateAgentBuilder = builder.AddAIAgent( "pirate", instructions: "You are a pirate. Speak like a pirate", description: "An agent that speaks like a pirate.", chatClientServiceKey: "chat-model") + .WithAITool(new CustomAITool()) + .WithAITool(new CustomFunctionTool()) .WithInMemoryThreadStore(); var knightsKnavesAgentBuilder = builder.AddAIAgent("knights-and-knaves", (sp, key) => @@ -78,11 +89,62 @@ Once the user has deduced what type (knight or knave) both Alice and Bob are, te description: "An agent that helps with literature.", chatClientServiceKey: "chat-model"); -builder.AddSequentialWorkflow("science-sequential-workflow", [chemistryAgent, mathsAgent, literatureAgent]).AddAsAIAgent(); -builder.AddConcurrentWorkflow("science-concurrent-workflow", [chemistryAgent, mathsAgent, literatureAgent]).AddAsAIAgent(); +var scienceSequentialWorkflow = builder.AddWorkflow("science-sequential-workflow", (sp, key) => +{ + List usedAgents = [chemistryAgent, mathsAgent, literatureAgent]; + var agents = usedAgents.Select(ab => sp.GetRequiredKeyedService(ab.Name)); + return AgentWorkflowBuilder.BuildSequential(workflowName: key, agents: agents); +}).AddAsAIAgent(); -builder.AddOpenAIChatCompletions(); -builder.AddOpenAIResponses(); +var scienceConcurrentWorkflow = builder.AddWorkflow("science-concurrent-workflow", (sp, key) => +{ + List usedAgents = [chemistryAgent, mathsAgent, literatureAgent]; + var agents = usedAgents.Select(ab => sp.GetRequiredKeyedService(ab.Name)); + return AgentWorkflowBuilder.BuildConcurrent(workflowName: key, agents: agents); +}).AddAsAIAgent(); + +builder.AddWorkflow("nonAgentWorkflow", (sp, key) => +{ + List usedAgents = [pirateAgentBuilder, chemistryAgent]; + var agents = usedAgents.Select(ab => sp.GetRequiredKeyedService(ab.Name)); + return AgentWorkflowBuilder.BuildSequential(workflowName: key, agents: agents); +}); + +builder.Services.AddKeyedSingleton("NonAgentAndNonmatchingDINameWorkflow", (sp, key) => +{ + List usedAgents = [pirateAgentBuilder, chemistryAgent]; + var agents = usedAgents.Select(ab => sp.GetRequiredKeyedService(ab.Name)); + return AgentWorkflowBuilder.BuildSequential(workflowName: "random-name", agents: agents); +}); + +builder.Services.AddSingleton(sp => +{ + var chatClient = sp.GetRequiredKeyedService("chat-model"); + return new ChatClientAgent(chatClient, name: "default-agent", instructions: "you are a default agent."); +}); + +builder.Services.AddKeyedSingleton("my-di-nonmatching-agent", (sp, name) => +{ + var chatClient = sp.GetRequiredKeyedService("chat-model"); + return new ChatClientAgent( + chatClient, + name: "some-random-name", // demonstrating registration can be different for DI and actual agent + instructions: "you are a dependency inject agent. Tell me all about dependency injection."); +}); + +builder.Services.AddKeyedSingleton("my-di-matchingname-agent", (sp, name) => +{ + if (name is not string nameStr) + { + throw new NotSupportedException("Name should be passed as a key"); + } + + var chatClient = sp.GetRequiredKeyedService("chat-model"); + return new ChatClientAgent( + chatClient, + name: nameStr, // demonstrating registration with the same name + instructions: "you are a dependency inject agent. Tell me all about dependency injection."); +}); var app = builder.Build(); @@ -93,8 +155,8 @@ Once the user has deduced what type (knight or knave) both Alice and Bob are, te app.UseExceptionHandler(); // attach a2a with simple message communication -app.MapA2A(agentName: "pirate", path: "/a2a/pirate"); -app.MapA2A(agentName: "knights-and-knaves", path: "/a2a/knights-and-knaves", agentCard: new() +app.MapA2A(pirateAgentBuilder, path: "/a2a/pirate"); +app.MapA2A(knightsKnavesAgentBuilder, path: "/a2a/knights-and-knaves", agentCard: new() { Name = "Knights and Knaves", Description = "An agent that helps you solve the knights and knaves puzzle.", @@ -104,7 +166,10 @@ Once the user has deduced what type (knight or knave) both Alice and Bob are, te // Url = "http://localhost:5390/a2a/knights-and-knaves" }); +app.MapDevUI(); + app.MapOpenAIResponses(); +app.MapOpenAIConversations(); app.MapOpenAIChatCompletions(pirateAgentBuilder); app.MapOpenAIChatCompletions(knightsKnavesAgentBuilder); diff --git a/dotnet/samples/AgentWebChat/AgentWebChat.AppHost/AgentWebChat.AppHost.csproj b/dotnet/samples/AgentWebChat/AgentWebChat.AppHost/AgentWebChat.AppHost.csproj index 464ba54db8..de87c119ec 100644 --- a/dotnet/samples/AgentWebChat/AgentWebChat.AppHost/AgentWebChat.AppHost.csproj +++ b/dotnet/samples/AgentWebChat/AgentWebChat.AppHost/AgentWebChat.AppHost.csproj @@ -4,7 +4,7 @@ Exe - net9.0 + net10.0 enable enable true diff --git a/dotnet/samples/AgentWebChat/AgentWebChat.AppHost/Program.cs b/dotnet/samples/AgentWebChat/AgentWebChat.AppHost/Program.cs index a28b3e1902..328e3f5e83 100644 --- a/dotnet/samples/AgentWebChat/AgentWebChat.AppHost/Program.cs +++ b/dotnet/samples/AgentWebChat/AgentWebChat.AppHost/Program.cs @@ -9,7 +9,9 @@ var chatModel = builder.AddAIModel("chat-model").AsAzureOpenAI("gpt-4o", o => o.AsExisting(azOpenAiResource, azOpenAiResourceGroup)); var agentHost = builder.AddProject("agenthost") - .WithReference(chatModel); + .WithHttpEndpoint(name: "devui") + .WithUrlForEndpoint("devui", (url) => new() { Url = "/devui", DisplayText = "Dev UI" }) + .WithReference(chatModel); builder.AddProject("webfrontend") .WithExternalHttpEndpoints() diff --git a/dotnet/samples/AgentWebChat/AgentWebChat.ServiceDefaults/AgentWebChat.ServiceDefaults.csproj b/dotnet/samples/AgentWebChat/AgentWebChat.ServiceDefaults/AgentWebChat.ServiceDefaults.csproj index 09110f11ad..0c5573beac 100644 --- a/dotnet/samples/AgentWebChat/AgentWebChat.ServiceDefaults/AgentWebChat.ServiceDefaults.csproj +++ b/dotnet/samples/AgentWebChat/AgentWebChat.ServiceDefaults/AgentWebChat.ServiceDefaults.csproj @@ -1,7 +1,7 @@ - net9.0 + net10.0 enable enable true diff --git a/dotnet/samples/AgentWebChat/AgentWebChat.Web/A2AAgentClient.cs b/dotnet/samples/AgentWebChat/AgentWebChat.Web/A2AAgentClient.cs index db690950da..08dafea129 100644 --- a/dotnet/samples/AgentWebChat/AgentWebChat.Web/A2AAgentClient.cs +++ b/dotnet/samples/AgentWebChat/AgentWebChat.Web/A2AAgentClient.cs @@ -25,7 +25,7 @@ public A2AAgentClient(ILogger logger, Uri baseUri) this._uri = baseUri; } - public async override IAsyncEnumerable RunStreamingAsync( + public override async IAsyncEnumerable RunStreamingAsync( string agentName, IList messages, string? threadId = null, @@ -122,7 +122,7 @@ public async override IAsyncEnumerable RunStreamingAsync } } - public async override Task GetAgentCardAsync(string agentName, CancellationToken cancellationToken = default) + public override async Task GetAgentCardAsync(string agentName, CancellationToken cancellationToken = default) { this._logger.LogInformation("Retrieving agent card for {Agent}", agentName); diff --git a/dotnet/samples/AgentWebChat/AgentWebChat.Web/AgentWebChat.Web.csproj b/dotnet/samples/AgentWebChat/AgentWebChat.Web/AgentWebChat.Web.csproj index 72541f046f..fd26f56191 100644 --- a/dotnet/samples/AgentWebChat/AgentWebChat.Web/AgentWebChat.Web.csproj +++ b/dotnet/samples/AgentWebChat/AgentWebChat.Web/AgentWebChat.Web.csproj @@ -1,7 +1,7 @@  - net9.0 + net10.0 enable enable $(NoWarn);CA1812 @@ -15,11 +15,4 @@
- - - - - - - diff --git a/dotnet/samples/AgentWebChat/AgentWebChat.Web/OpenAIChatCompletionsAgentClient.cs b/dotnet/samples/AgentWebChat/AgentWebChat.Web/OpenAIChatCompletionsAgentClient.cs index ae71a87678..95e3d16fd4 100644 --- a/dotnet/samples/AgentWebChat/AgentWebChat.Web/OpenAIChatCompletionsAgentClient.cs +++ b/dotnet/samples/AgentWebChat/AgentWebChat.Web/OpenAIChatCompletionsAgentClient.cs @@ -16,7 +16,7 @@ namespace AgentWebChat.Web; /// internal sealed class OpenAIChatCompletionsAgentClient(HttpClient httpClient) : AgentClientBase { - public async override IAsyncEnumerable RunStreamingAsync( + public override async IAsyncEnumerable RunStreamingAsync( string agentName, IList messages, string? threadId = null, diff --git a/dotnet/samples/AgentWebChat/AgentWebChat.Web/OpenAIResponsesAgentClient.cs b/dotnet/samples/AgentWebChat/AgentWebChat.Web/OpenAIResponsesAgentClient.cs index bb7f6c151c..7cc85b97c3 100644 --- a/dotnet/samples/AgentWebChat/AgentWebChat.Web/OpenAIResponsesAgentClient.cs +++ b/dotnet/samples/AgentWebChat/AgentWebChat.Web/OpenAIResponsesAgentClient.cs @@ -15,7 +15,7 @@ namespace AgentWebChat.Web; /// internal sealed class OpenAIResponsesAgentClient(HttpClient httpClient) : AgentClientBase { - public async override IAsyncEnumerable RunStreamingAsync( + public override async IAsyncEnumerable RunStreamingAsync( string agentName, IList messages, string? threadId = null, diff --git a/dotnet/samples/AzureFunctions/.editorconfig b/dotnet/samples/AzureFunctions/.editorconfig new file mode 100644 index 0000000000..b43bf5ebd0 --- /dev/null +++ b/dotnet/samples/AzureFunctions/.editorconfig @@ -0,0 +1,10 @@ +# .editorconfig +[*.cs] + +# See https://github.com/Azure/azure-functions-durable-extension/issues/3173 +dotnet_diagnostic.DURABLE0001.severity = none +dotnet_diagnostic.DURABLE0002.severity = none +dotnet_diagnostic.DURABLE0003.severity = none +dotnet_diagnostic.DURABLE0004.severity = none +dotnet_diagnostic.DURABLE0005.severity = none +dotnet_diagnostic.DURABLE0006.severity = none diff --git a/dotnet/samples/AzureFunctions/01_SingleAgent/01_SingleAgent.csproj b/dotnet/samples/AzureFunctions/01_SingleAgent/01_SingleAgent.csproj new file mode 100644 index 0000000000..99f78cc1ab --- /dev/null +++ b/dotnet/samples/AzureFunctions/01_SingleAgent/01_SingleAgent.csproj @@ -0,0 +1,42 @@ + + + net10.0 + v4 + Exe + enable + enable + + SingleAgent + SingleAgent + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/dotnet/samples/AzureFunctions/01_SingleAgent/Program.cs b/dotnet/samples/AzureFunctions/01_SingleAgent/Program.cs new file mode 100644 index 0000000000..b3d40a120c --- /dev/null +++ b/dotnet/samples/AzureFunctions/01_SingleAgent/Program.cs @@ -0,0 +1,37 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Azure; +using Azure.AI.OpenAI; +using Azure.Identity; +using Microsoft.Agents.AI; +using Microsoft.Agents.AI.Hosting.AzureFunctions; +using Microsoft.Azure.Functions.Worker.Builder; +using Microsoft.Extensions.Hosting; +using OpenAI.Chat; + +// Get the Azure OpenAI endpoint and deployment name from environment variables. +string endpoint = Environment.GetEnvironmentVariable("AZURE_OPENAI_ENDPOINT") + ?? throw new InvalidOperationException("AZURE_OPENAI_ENDPOINT is not set."); +string deploymentName = Environment.GetEnvironmentVariable("AZURE_OPENAI_DEPLOYMENT") + ?? throw new InvalidOperationException("AZURE_OPENAI_DEPLOYMENT is not set."); + +// Use Azure Key Credential if provided, otherwise use Azure CLI Credential. +string? azureOpenAiKey = System.Environment.GetEnvironmentVariable("AZURE_OPENAI_KEY"); +AzureOpenAIClient client = !string.IsNullOrEmpty(azureOpenAiKey) + ? new AzureOpenAIClient(new Uri(endpoint), new AzureKeyCredential(azureOpenAiKey)) + : new AzureOpenAIClient(new Uri(endpoint), new AzureCliCredential()); + +// Set up an AI agent following the standard Microsoft Agent Framework pattern. +const string JokerName = "Joker"; +const string JokerInstructions = "You are good at telling jokes."; + +AIAgent agent = client.GetChatClient(deploymentName).CreateAIAgent(JokerInstructions, JokerName); + +// Configure the function app to host the AI agent. +// This will automatically generate HTTP API endpoints for the agent. +using IHost app = FunctionsApplication + .CreateBuilder(args) + .ConfigureFunctionsWebApplication() + .ConfigureDurableAgents(options => options.AddAIAgent(agent)) + .Build(); +app.Run(); diff --git a/dotnet/samples/AzureFunctions/01_SingleAgent/README.md b/dotnet/samples/AzureFunctions/01_SingleAgent/README.md new file mode 100644 index 0000000000..d4ac968978 --- /dev/null +++ b/dotnet/samples/AzureFunctions/01_SingleAgent/README.md @@ -0,0 +1,89 @@ +# Single Agent Sample + +This sample demonstrates how to use the Durable Agent Framework (DAFx) to create a simple Azure Functions app that hosts a single AI agent and provides direct HTTP API access for interactive conversations. + +## Key Concepts Demonstrated + +- Using the Microsoft Agent Framework to define a simple AI agent with a name and instructions. +- Registering agents with the Function app and running them using HTTP. +- Conversation management (via session IDs) for isolated interactions. + +## Environment Setup + +See the [README.md](../README.md) file in the parent directory for more information on how to configure the environment, including how to install and run common sample dependencies. + +## Running the Sample + +With the environment setup and function app running, you can test the sample by sending an HTTP request to the agent endpoint. + +You can use the `demo.http` file to send a message to the agent, or a command line tool like `curl` as shown below: + +Bash (Linux/macOS/WSL): + +```bash +curl -X POST http://localhost:7071/api/agents/Joker/run \ + -H "Content-Type: text/plain" \ + -d "Tell me a joke about a pirate." +``` + +PowerShell: + +```powershell +Invoke-RestMethod -Method Post ` + -Uri http://localhost:7071/api/agents/Joker/run ` + -ContentType text/plain ` + -Body "Tell me a joke about a pirate." +``` + +You can also send JSON requests: + +```bash +curl -X POST http://localhost:7071/api/agents/Joker/run \ + -H "Content-Type: application/json" \ + -H "Accept: application/json" \ + -d '{"message": "Tell me a joke about a pirate."}' +``` + +To continue a conversation, include the `thread_id` in the query string or JSON body: + +```bash +curl -X POST "http://localhost:7071/api/agents/Joker/run?thread_id=your-thread-id" \ + -H "Content-Type: application/json" \ + -H "Accept: application/json" \ + -d '{"message": "Tell me another one."}' +``` + +The response from the agent will be displayed in the terminal where you ran `func start`. The expected `text/plain` output will look something like: + +```text +Why don't pirates ever learn the alphabet? Because they always get stuck at "C"! +``` + +The expected `application/json` output will look something like: + +```json +{ + "status": 200, + "thread_id": "ee6e47a0-f24b-40b1-ade8-16fcebb9eb40", + "response": { + "Messages": [ + { + "AuthorName": "Joker", + "CreatedAt": "2025-11-11T12:00:00.0000000Z", + "Role": "assistant", + "Contents": [ + { + "Type": "text", + "Text": "Why don't pirates ever learn the alphabet? Because they always get stuck at 'C'!" + } + ] + } + ], + "Usage": { + "InputTokenCount": 78, + "OutputTokenCount": 36, + "TotalTokenCount": 114 + } + } +} +``` diff --git a/dotnet/samples/AzureFunctions/01_SingleAgent/demo.http b/dotnet/samples/AzureFunctions/01_SingleAgent/demo.http new file mode 100644 index 0000000000..3b741adf31 --- /dev/null +++ b/dotnet/samples/AzureFunctions/01_SingleAgent/demo.http @@ -0,0 +1,8 @@ +# Default endpoint address for local testing +@authority=http://localhost:7071 + +### Prompt the agent +POST {{authority}}/api/agents/Joker/run +Content-Type: text/plain + +Tell me a joke about a pirate. diff --git a/dotnet/samples/AzureFunctions/01_SingleAgent/host.json b/dotnet/samples/AzureFunctions/01_SingleAgent/host.json new file mode 100644 index 0000000000..9384a0a583 --- /dev/null +++ b/dotnet/samples/AzureFunctions/01_SingleAgent/host.json @@ -0,0 +1,20 @@ +{ + "version": "2.0", + "logging": { + "logLevel": { + "Microsoft.Agents.AI.DurableTask": "Information", + "Microsoft.Agents.AI.Hosting.AzureFunctions": "Information", + "DurableTask": "Information", + "Microsoft.DurableTask": "Information" + } + }, + "extensions": { + "durableTask": { + "hubName": "default", + "storageProvider": { + "type": "AzureManaged", + "connectionStringName": "DURABLE_TASK_SCHEDULER_CONNECTION_STRING" + } + } + } +} diff --git a/dotnet/samples/AzureFunctions/01_SingleAgent/local.settings.json b/dotnet/samples/AzureFunctions/01_SingleAgent/local.settings.json new file mode 100644 index 0000000000..3411463ac4 --- /dev/null +++ b/dotnet/samples/AzureFunctions/01_SingleAgent/local.settings.json @@ -0,0 +1,10 @@ +{ + "IsEncrypted": false, + "Values": { + "FUNCTIONS_WORKER_RUNTIME": "dotnet-isolated", + "AzureWebJobsStorage": "UseDevelopmentStorage=true", + "DURABLE_TASK_SCHEDULER_CONNECTION_STRING": "Endpoint=http://localhost:8080;TaskHub=default;Authentication=None", + "AZURE_OPENAI_ENDPOINT": "", + "AZURE_OPENAI_DEPLOYMENT": "" + } +} \ No newline at end of file diff --git a/dotnet/samples/AzureFunctions/02_AgentOrchestration_Chaining/02_AgentOrchestration_Chaining.csproj b/dotnet/samples/AzureFunctions/02_AgentOrchestration_Chaining/02_AgentOrchestration_Chaining.csproj new file mode 100644 index 0000000000..af6fe8bcde --- /dev/null +++ b/dotnet/samples/AzureFunctions/02_AgentOrchestration_Chaining/02_AgentOrchestration_Chaining.csproj @@ -0,0 +1,42 @@ + + + net10.0 + v4 + Exe + enable + enable + + AgentOrchestration_Chaining + AgentOrchestration_Chaining + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/dotnet/samples/AzureFunctions/02_AgentOrchestration_Chaining/FunctionTriggers.cs b/dotnet/samples/AzureFunctions/02_AgentOrchestration_Chaining/FunctionTriggers.cs new file mode 100644 index 0000000000..a631e7715c --- /dev/null +++ b/dotnet/samples/AzureFunctions/02_AgentOrchestration_Chaining/FunctionTriggers.cs @@ -0,0 +1,92 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Net; +using System.Text.Json; +using Microsoft.Agents.AI; +using Microsoft.Agents.AI.DurableTask; +using Microsoft.Azure.Functions.Worker; +using Microsoft.Azure.Functions.Worker.Http; +using Microsoft.DurableTask; +using Microsoft.DurableTask.Client; + +namespace AgentOrchestration_Chaining; + +public static class FunctionTriggers +{ + public sealed record TextResponse(string Text); + + [Function(nameof(RunOrchestrationAsync))] + public static async Task RunOrchestrationAsync([OrchestrationTrigger] TaskOrchestrationContext context) + { + DurableAIAgent writer = context.GetAgent("WriterAgent"); + AgentThread writerThread = writer.GetNewThread(); + + AgentRunResponse initial = await writer.RunAsync( + message: "Write a concise inspirational sentence about learning.", + thread: writerThread); + + AgentRunResponse refined = await writer.RunAsync( + message: $"Improve this further while keeping it under 25 words: {initial.Result.Text}", + thread: writerThread); + + return refined.Result.Text; + } + + // POST /singleagent/run + [Function(nameof(StartOrchestrationAsync))] + public static async Task StartOrchestrationAsync( + [HttpTrigger(AuthorizationLevel.Anonymous, "post", Route = "singleagent/run")] HttpRequestData req, + [DurableClient] DurableTaskClient client) + { + string instanceId = await client.ScheduleNewOrchestrationInstanceAsync( + orchestratorName: nameof(RunOrchestrationAsync)); + + HttpResponseData response = req.CreateResponse(HttpStatusCode.Accepted); + await response.WriteAsJsonAsync(new + { + message = "Single-agent orchestration started.", + instanceId, + statusQueryGetUri = GetStatusQueryGetUri(req, instanceId), + }); + return response; + } + + // GET /singleagent/status/{instanceId} + [Function(nameof(GetOrchestrationStatusAsync))] + public static async Task GetOrchestrationStatusAsync( + [HttpTrigger(AuthorizationLevel.Anonymous, "get", Route = "singleagent/status/{instanceId}")] HttpRequestData req, + string instanceId, + [DurableClient] DurableTaskClient client) + { + OrchestrationMetadata? status = await client.GetInstanceAsync( + instanceId, + getInputsAndOutputs: true, + req.FunctionContext.CancellationToken); + + if (status is null) + { + HttpResponseData notFound = req.CreateResponse(HttpStatusCode.NotFound); + await notFound.WriteAsJsonAsync(new { error = "Instance not found" }); + return notFound; + } + + HttpResponseData response = req.CreateResponse(HttpStatusCode.OK); + await response.WriteAsJsonAsync(new + { + instanceId = status.InstanceId, + runtimeStatus = status.RuntimeStatus.ToString(), + input = status.SerializedInput is not null ? (object)status.ReadInputAs() : null, + output = status.SerializedOutput is not null ? (object)status.ReadOutputAs() : null, + failureDetails = status.FailureDetails + }); + return response; + } + + private static string GetStatusQueryGetUri(HttpRequestData req, string instanceId) + { + // NOTE: This can be made more robust by considering the value of + // request headers like "X-Forwarded-Host" and "X-Forwarded-Proto". + string authority = $"{req.Url.Scheme}://{req.Url.Authority}"; + return $"{authority}/api/singleagent/status/{instanceId}"; + } +} diff --git a/dotnet/samples/AzureFunctions/02_AgentOrchestration_Chaining/Program.cs b/dotnet/samples/AzureFunctions/02_AgentOrchestration_Chaining/Program.cs new file mode 100644 index 0000000000..3776ecd062 --- /dev/null +++ b/dotnet/samples/AzureFunctions/02_AgentOrchestration_Chaining/Program.cs @@ -0,0 +1,40 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Azure; +using Azure.AI.OpenAI; +using Azure.Identity; +using Microsoft.Agents.AI; +using Microsoft.Agents.AI.Hosting.AzureFunctions; +using Microsoft.Azure.Functions.Worker.Builder; +using Microsoft.Extensions.Hosting; +using OpenAI.Chat; + +// Get the Azure OpenAI endpoint and deployment name from environment variables. +string endpoint = Environment.GetEnvironmentVariable("AZURE_OPENAI_ENDPOINT") + ?? throw new InvalidOperationException("AZURE_OPENAI_ENDPOINT is not set."); +string deploymentName = Environment.GetEnvironmentVariable("AZURE_OPENAI_DEPLOYMENT") + ?? throw new InvalidOperationException("AZURE_OPENAI_DEPLOYMENT is not set."); + +// Use Azure Key Credential if provided, otherwise use Azure CLI Credential. +string? azureOpenAiKey = System.Environment.GetEnvironmentVariable("AZURE_OPENAI_KEY"); +AzureOpenAIClient client = !string.IsNullOrEmpty(azureOpenAiKey) + ? new AzureOpenAIClient(new Uri(endpoint), new AzureKeyCredential(azureOpenAiKey)) + : new AzureOpenAIClient(new Uri(endpoint), new AzureCliCredential()); + +// Single agent used by the orchestration to demonstrate sequential calls on the same thread. +const string WriterName = "WriterAgent"; +const string WriterInstructions = + """ + You refine short pieces of text. When given an initial sentence you enhance it; + when given an improved sentence you polish it further. + """; + +AIAgent writerAgent = client.GetChatClient(deploymentName).CreateAIAgent(WriterInstructions, WriterName); + +using IHost app = FunctionsApplication + .CreateBuilder(args) + .ConfigureFunctionsWebApplication() + .ConfigureDurableAgents(options => options.AddAIAgent(writerAgent)) + .Build(); + +app.Run(); diff --git a/dotnet/samples/AzureFunctions/02_AgentOrchestration_Chaining/README.md b/dotnet/samples/AzureFunctions/02_AgentOrchestration_Chaining/README.md new file mode 100644 index 0000000000..e98885eced --- /dev/null +++ b/dotnet/samples/AzureFunctions/02_AgentOrchestration_Chaining/README.md @@ -0,0 +1,59 @@ +# Single Agent Orchestration Sample + +This sample demonstrates how to use the Durable Agent Framework (DAFx) to create a simple Azure Functions app that orchestrates sequential calls to a single AI agent using the same conversation thread for context continuity. + +## Key Concepts Demonstrated + +- Orchestrating multiple interactions with the same agent in a deterministic order +- Using the same `AgentThread` across multiple calls to maintain conversational context +- Durable orchestration with automatic checkpointing and resumption from failures +- HTTP API integration for starting and monitoring orchestrations + +## Environment Setup + +See the [README.md](../README.md) file in the parent directory for more information on how to configure the environment, including how to install and run common sample dependencies. + +## Running the Sample + +With the environment setup and function app running, you can test the sample by sending an HTTP request to start the orchestration. + +You can use the `demo.http` file to start the orchestration, or a command line tool like `curl` as shown below: + +Bash (Linux/macOS/WSL): + +```bash +curl -X POST http://localhost:7071/api/singleagent/run +``` + +PowerShell: + +```powershell +Invoke-RestMethod -Method Post -Uri http://localhost:7071/api/singleagent/run +``` + +The response will be a JSON object that looks something like the following, which indicates that the orchestration has started. + +```json +{ + "message": "Single-agent orchestration started.", + "instanceId": "86313f1d45fb42eeb50b1852626bf3ff", + "statusQueryGetUri": "http://localhost:7071/api/singleagent/status/86313f1d45fb42eeb50b1852626bf3ff" +} +``` + +The orchestration will proceed to run the WriterAgent twice in sequence: + +1. First, it writes an inspirational sentence about learning +2. Then, it refines the initial output using the same conversation thread + +Once the orchestration has completed, you can get the status of the orchestration by sending a GET request to the `statusQueryGetUri` URL. The response will be a JSON object that looks something like the following: + +```json +{ + "failureDetails": null, + "input": null, + "instanceId": "86313f1d45fb42eeb50b1852626bf3ff", + "output": "Learning serves as the key, opening doors to boundless opportunities and a brighter future.", + "runtimeStatus": "Completed" +} +``` diff --git a/dotnet/samples/AzureFunctions/02_AgentOrchestration_Chaining/demo.http b/dotnet/samples/AzureFunctions/02_AgentOrchestration_Chaining/demo.http new file mode 100644 index 0000000000..aa4dcc4a16 --- /dev/null +++ b/dotnet/samples/AzureFunctions/02_AgentOrchestration_Chaining/demo.http @@ -0,0 +1,3 @@ +### Start the single-agent orchestration +POST http://localhost:7071/api/singleagent/run + diff --git a/dotnet/samples/AzureFunctions/02_AgentOrchestration_Chaining/host.json b/dotnet/samples/AzureFunctions/02_AgentOrchestration_Chaining/host.json new file mode 100644 index 0000000000..9384a0a583 --- /dev/null +++ b/dotnet/samples/AzureFunctions/02_AgentOrchestration_Chaining/host.json @@ -0,0 +1,20 @@ +{ + "version": "2.0", + "logging": { + "logLevel": { + "Microsoft.Agents.AI.DurableTask": "Information", + "Microsoft.Agents.AI.Hosting.AzureFunctions": "Information", + "DurableTask": "Information", + "Microsoft.DurableTask": "Information" + } + }, + "extensions": { + "durableTask": { + "hubName": "default", + "storageProvider": { + "type": "AzureManaged", + "connectionStringName": "DURABLE_TASK_SCHEDULER_CONNECTION_STRING" + } + } + } +} diff --git a/dotnet/samples/AzureFunctions/02_AgentOrchestration_Chaining/local.settings.json b/dotnet/samples/AzureFunctions/02_AgentOrchestration_Chaining/local.settings.json new file mode 100644 index 0000000000..54dfbb5664 --- /dev/null +++ b/dotnet/samples/AzureFunctions/02_AgentOrchestration_Chaining/local.settings.json @@ -0,0 +1,10 @@ +{ + "IsEncrypted": false, + "Values": { + "FUNCTIONS_WORKER_RUNTIME": "dotnet-isolated", + "AzureWebJobsStorage": "UseDevelopmentStorage=true", + "DURABLE_TASK_SCHEDULER_CONNECTION_STRING": "Endpoint=http://localhost:8080;TaskHub=default;Authentication=None", + "AZURE_OPENAI_ENDPOINT": "", + "AZURE_OPENAI_DEPLOYMENT": "" + } +} diff --git a/dotnet/samples/AzureFunctions/03_AgentOrchestration_Concurrency/03_AgentOrchestration_Concurrency.csproj b/dotnet/samples/AzureFunctions/03_AgentOrchestration_Concurrency/03_AgentOrchestration_Concurrency.csproj new file mode 100644 index 0000000000..394bf9cc35 --- /dev/null +++ b/dotnet/samples/AzureFunctions/03_AgentOrchestration_Concurrency/03_AgentOrchestration_Concurrency.csproj @@ -0,0 +1,42 @@ + + + net10.0 + v4 + Exe + enable + enable + + AgentOrchestration_Concurrency + AgentOrchestration_Concurrency + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/dotnet/samples/AzureFunctions/03_AgentOrchestration_Concurrency/FunctionTriggers.cs b/dotnet/samples/AzureFunctions/03_AgentOrchestration_Concurrency/FunctionTriggers.cs new file mode 100644 index 0000000000..2d15dd585c --- /dev/null +++ b/dotnet/samples/AzureFunctions/03_AgentOrchestration_Concurrency/FunctionTriggers.cs @@ -0,0 +1,116 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Net; +using System.Text.Json; +using Microsoft.Agents.AI; +using Microsoft.Agents.AI.DurableTask; +using Microsoft.Azure.Functions.Worker; +using Microsoft.Azure.Functions.Worker.Http; +using Microsoft.DurableTask; +using Microsoft.DurableTask.Client; + +namespace AgentOrchestration_Concurrency; + +public static class FunctionsTriggers +{ + public sealed record TextResponse(string Text); + + [Function(nameof(RunOrchestrationAsync))] + public static async Task RunOrchestrationAsync([OrchestrationTrigger] TaskOrchestrationContext context) + { + // Get the prompt from the orchestration input + string prompt = context.GetInput() ?? throw new InvalidOperationException("Prompt is required"); + + // Get both agents + DurableAIAgent physicist = context.GetAgent("PhysicistAgent"); + DurableAIAgent chemist = context.GetAgent("ChemistAgent"); + + // Start both agent runs concurrently + Task> physicistTask = physicist.RunAsync(prompt); + + Task> chemistTask = chemist.RunAsync(prompt); + + // Wait for both tasks to complete using Task.WhenAll + await Task.WhenAll(physicistTask, chemistTask); + + // Get the results + TextResponse physicistResponse = (await physicistTask).Result; + TextResponse chemistResponse = (await chemistTask).Result; + + // Return the result as a structured, anonymous type + return new + { + physicist = physicistResponse.Text, + chemist = chemistResponse.Text, + }; + } + + // POST /multiagent/run + [Function(nameof(StartOrchestrationAsync))] + public static async Task StartOrchestrationAsync( + [HttpTrigger(AuthorizationLevel.Anonymous, "post", Route = "multiagent/run")] HttpRequestData req, + [DurableClient] DurableTaskClient client) + { + // Read the prompt from the request body + string? prompt = await req.ReadAsStringAsync(); + if (string.IsNullOrWhiteSpace(prompt)) + { + HttpResponseData badRequestResponse = req.CreateResponse(HttpStatusCode.BadRequest); + await badRequestResponse.WriteAsJsonAsync(new { error = "Prompt is required" }); + return badRequestResponse; + } + + string instanceId = await client.ScheduleNewOrchestrationInstanceAsync( + orchestratorName: nameof(RunOrchestrationAsync), + input: prompt); + + HttpResponseData response = req.CreateResponse(HttpStatusCode.Accepted); + await response.WriteAsJsonAsync(new + { + message = "Multi-agent concurrent orchestration started.", + prompt, + instanceId, + statusQueryGetUri = GetStatusQueryGetUri(req, instanceId), + }); + return response; + } + + // GET /multiagent/status/{instanceId} + [Function(nameof(GetOrchestrationStatusAsync))] + public static async Task GetOrchestrationStatusAsync( + [HttpTrigger(AuthorizationLevel.Anonymous, "get", Route = "multiagent/status/{instanceId}")] HttpRequestData req, + string instanceId, + [DurableClient] DurableTaskClient client) + { + OrchestrationMetadata? status = await client.GetInstanceAsync( + instanceId, + getInputsAndOutputs: true, + req.FunctionContext.CancellationToken); + + if (status is null) + { + HttpResponseData notFound = req.CreateResponse(HttpStatusCode.NotFound); + await notFound.WriteAsJsonAsync(new { error = "Instance not found" }); + return notFound; + } + + HttpResponseData response = req.CreateResponse(HttpStatusCode.OK); + await response.WriteAsJsonAsync(new + { + instanceId = status.InstanceId, + runtimeStatus = status.RuntimeStatus.ToString(), + input = status.SerializedInput is not null ? (object)status.ReadInputAs() : null, + output = status.SerializedOutput is not null ? (object)status.ReadOutputAs() : null, + failureDetails = status.FailureDetails + }); + return response; + } + + private static string GetStatusQueryGetUri(HttpRequestData req, string instanceId) + { + // NOTE: This can be made more robust by considering the value of + // request headers like "X-Forwarded-Host" and "X-Forwarded-Proto". + string authority = $"{req.Url.Scheme}://{req.Url.Authority}"; + return $"{authority}/api/multiagent/status/{instanceId}"; + } +} diff --git a/dotnet/samples/AzureFunctions/03_AgentOrchestration_Concurrency/Program.cs b/dotnet/samples/AzureFunctions/03_AgentOrchestration_Concurrency/Program.cs new file mode 100644 index 0000000000..dfc2049d45 --- /dev/null +++ b/dotnet/samples/AzureFunctions/03_AgentOrchestration_Concurrency/Program.cs @@ -0,0 +1,45 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Azure; +using Azure.AI.OpenAI; +using Azure.Identity; +using Microsoft.Agents.AI; +using Microsoft.Agents.AI.Hosting.AzureFunctions; +using Microsoft.Azure.Functions.Worker.Builder; +using Microsoft.Extensions.Hosting; +using OpenAI.Chat; + +// Get the Azure OpenAI endpoint and deployment name from environment variables. +string endpoint = Environment.GetEnvironmentVariable("AZURE_OPENAI_ENDPOINT") + ?? throw new InvalidOperationException("AZURE_OPENAI_ENDPOINT is not set."); +string deploymentName = Environment.GetEnvironmentVariable("AZURE_OPENAI_DEPLOYMENT") + ?? throw new InvalidOperationException("AZURE_OPENAI_DEPLOYMENT is not set."); + +// Use Azure Key Credential if provided, otherwise use Azure CLI Credential. +string? azureOpenAiKey = System.Environment.GetEnvironmentVariable("AZURE_OPENAI_KEY"); +AzureOpenAIClient client = !string.IsNullOrEmpty(azureOpenAiKey) + ? new AzureOpenAIClient(new Uri(endpoint), new AzureKeyCredential(azureOpenAiKey)) + : new AzureOpenAIClient(new Uri(endpoint), new AzureCliCredential()); + +// Two agents used by the orchestration to demonstrate concurrent execution. +const string PhysicistName = "PhysicistAgent"; +const string PhysicistInstructions = "You are an expert in physics. You answer questions from a physics perspective."; + +const string ChemistName = "ChemistAgent"; +const string ChemistInstructions = "You are an expert in chemistry. You answer questions from a chemistry perspective."; + +AIAgent physicistAgent = client.GetChatClient(deploymentName).CreateAIAgent(PhysicistInstructions, PhysicistName); +AIAgent chemistAgent = client.GetChatClient(deploymentName).CreateAIAgent(ChemistInstructions, ChemistName); + +using IHost app = FunctionsApplication + .CreateBuilder(args) + .ConfigureFunctionsWebApplication() + .ConfigureDurableAgents(options => + { + options + .AddAIAgent(physicistAgent) + .AddAIAgent(chemistAgent); + }) + .Build(); + +app.Run(); diff --git a/dotnet/samples/AzureFunctions/03_AgentOrchestration_Concurrency/README.md b/dotnet/samples/AzureFunctions/03_AgentOrchestration_Concurrency/README.md new file mode 100644 index 0000000000..974aa1f2d2 --- /dev/null +++ b/dotnet/samples/AzureFunctions/03_AgentOrchestration_Concurrency/README.md @@ -0,0 +1,65 @@ +# Multi-Agent Concurrent Orchestration Sample + +This sample demonstrates how to use the Durable Agent Framework (DAFx) to create an Azure Functions app that orchestrates concurrent execution of multiple AI agents, each with specialized expertise, to provide comprehensive answers to complex questions. + +## Key Concepts Demonstrated + +- Multi-agent orchestration with specialized AI agents (physics and chemistry) +- Concurrent execution using the fan-out/fan-in pattern for improved performance and distributed processing +- Response aggregation from multiple agents into a unified result +- Durable orchestration with automatic checkpointing and resumption from failures + +## Environment Setup + +See the [README.md](../README.md) file in the parent directory for more information on how to configure the environment, including how to install and run common sample dependencies. + +## Running the Sample + +With the environment setup and function app running, you can test the sample by sending an HTTP request with a custom prompt to the orchestration. + +You can use the `demo.http` file to send a message to the agents, or a command line tool like `curl` as shown below: + +Bash (Linux/macOS/WSL): + +```bash +curl -X POST http://localhost:7071/api/multiagent/run \ + -H "Content-Type: text/plain" \ + -d "What is temperature?" +``` + +PowerShell: + +```powershell +Invoke-RestMethod -Method Post ` + -Uri http://localhost:7071/api/multiagent/run ` + -ContentType text/plain ` + -Body "What is temperature?" +``` + +The response will be a JSON object that looks something like the following, which indicates that the orchestration has started. + +```json +{ + "message": "Multi-agent concurrent orchestration started.", + "prompt": "What is temperature?", + "instanceId": "e7e29999b6b8424682b3539292afc9ed", + "statusQueryGetUri": "http://localhost:7071/api/multiagent/status/e7e29999b6b8424682b3539292afc9ed" +} +``` + +The orchestration will run both the PhysicistAgent and ChemistAgent concurrently, asking them the same question. Their responses will be combined to provide a comprehensive answer covering both physical and chemical aspects. + +Once the orchestration has completed, you can get the status of the orchestration by sending a GET request to the `statusQueryGetUri` URL. The response will be a JSON object that looks something like the following: + +```json +{ + "failureDetails": null, + "input": "What is temperature?", + "instanceId": "e7e29999b6b8424682b3539292afc9ed", + "output": { + "physicist": "Temperature is a measure of the average kinetic energy of particles in a system. From a physics perspective, it represents the thermal energy and determines the direction of heat flow between objects.", + "chemist": "From a chemistry perspective, temperature is crucial for chemical reactions as it affects reaction rates through the Arrhenius equation. It influences the equilibrium position of reversible reactions and determines the physical state of substances." + }, + "runtimeStatus": "Completed" +} +``` diff --git a/dotnet/samples/AzureFunctions/03_AgentOrchestration_Concurrency/demo.http b/dotnet/samples/AzureFunctions/03_AgentOrchestration_Concurrency/demo.http new file mode 100644 index 0000000000..8004e27e8e --- /dev/null +++ b/dotnet/samples/AzureFunctions/03_AgentOrchestration_Concurrency/demo.http @@ -0,0 +1,5 @@ +### Start the multi-agent concurrent orchestration +POST http://localhost:7071/api/multiagent/run +Content-Type: text/plain + +What is temperature? diff --git a/dotnet/samples/AzureFunctions/03_AgentOrchestration_Concurrency/host.json b/dotnet/samples/AzureFunctions/03_AgentOrchestration_Concurrency/host.json new file mode 100644 index 0000000000..9384a0a583 --- /dev/null +++ b/dotnet/samples/AzureFunctions/03_AgentOrchestration_Concurrency/host.json @@ -0,0 +1,20 @@ +{ + "version": "2.0", + "logging": { + "logLevel": { + "Microsoft.Agents.AI.DurableTask": "Information", + "Microsoft.Agents.AI.Hosting.AzureFunctions": "Information", + "DurableTask": "Information", + "Microsoft.DurableTask": "Information" + } + }, + "extensions": { + "durableTask": { + "hubName": "default", + "storageProvider": { + "type": "AzureManaged", + "connectionStringName": "DURABLE_TASK_SCHEDULER_CONNECTION_STRING" + } + } + } +} diff --git a/dotnet/samples/AzureFunctions/03_AgentOrchestration_Concurrency/local.settings.json b/dotnet/samples/AzureFunctions/03_AgentOrchestration_Concurrency/local.settings.json new file mode 100644 index 0000000000..54dfbb5664 --- /dev/null +++ b/dotnet/samples/AzureFunctions/03_AgentOrchestration_Concurrency/local.settings.json @@ -0,0 +1,10 @@ +{ + "IsEncrypted": false, + "Values": { + "FUNCTIONS_WORKER_RUNTIME": "dotnet-isolated", + "AzureWebJobsStorage": "UseDevelopmentStorage=true", + "DURABLE_TASK_SCHEDULER_CONNECTION_STRING": "Endpoint=http://localhost:8080;TaskHub=default;Authentication=None", + "AZURE_OPENAI_ENDPOINT": "", + "AZURE_OPENAI_DEPLOYMENT": "" + } +} diff --git a/dotnet/samples/AzureFunctions/04_AgentOrchestration_Conditionals/04_AgentOrchestration_Conditionals.csproj b/dotnet/samples/AzureFunctions/04_AgentOrchestration_Conditionals/04_AgentOrchestration_Conditionals.csproj new file mode 100644 index 0000000000..8dc1832227 --- /dev/null +++ b/dotnet/samples/AzureFunctions/04_AgentOrchestration_Conditionals/04_AgentOrchestration_Conditionals.csproj @@ -0,0 +1,42 @@ + + + net10.0 + v4 + Exe + enable + enable + + AgentOrchestration_Conditionals + AgentOrchestration_Conditionals + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/dotnet/samples/AzureFunctions/04_AgentOrchestration_Conditionals/FunctionTriggers.cs b/dotnet/samples/AzureFunctions/04_AgentOrchestration_Conditionals/FunctionTriggers.cs new file mode 100644 index 0000000000..14a91185f8 --- /dev/null +++ b/dotnet/samples/AzureFunctions/04_AgentOrchestration_Conditionals/FunctionTriggers.cs @@ -0,0 +1,143 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Net; +using System.Text.Json; +using Microsoft.Agents.AI; +using Microsoft.Agents.AI.DurableTask; +using Microsoft.Azure.Functions.Worker; +using Microsoft.Azure.Functions.Worker.Http; +using Microsoft.DurableTask; +using Microsoft.DurableTask.Client; + +namespace AgentOrchestration_Conditionals; + +public static class FunctionTriggers +{ + [Function(nameof(RunOrchestrationAsync))] + public static async Task RunOrchestrationAsync([OrchestrationTrigger] TaskOrchestrationContext context) + { + // Get the email from the orchestration input + Email email = context.GetInput() ?? throw new InvalidOperationException("Email is required"); + + // Get the spam detection agent + DurableAIAgent spamDetectionAgent = context.GetAgent("SpamDetectionAgent"); + AgentThread spamThread = spamDetectionAgent.GetNewThread(); + + // Step 1: Check if the email is spam + AgentRunResponse spamDetectionResponse = await spamDetectionAgent.RunAsync( + message: + $""" + Analyze this email for spam content and return a JSON response with 'is_spam' (boolean) and 'reason' (string) fields: + Email ID: {email.EmailId} + Content: {email.EmailContent} + """, + thread: spamThread); + DetectionResult result = spamDetectionResponse.Result; + + // Step 2: Conditional logic based on spam detection result + if (result.IsSpam) + { + // Handle spam email + return await context.CallActivityAsync(nameof(HandleSpamEmail), result.Reason); + } + + // Generate and send response for legitimate email + DurableAIAgent emailAssistantAgent = context.GetAgent("EmailAssistantAgent"); + AgentThread emailThread = emailAssistantAgent.GetNewThread(); + + AgentRunResponse emailAssistantResponse = await emailAssistantAgent.RunAsync( + message: + $""" + Draft a professional response to this email. Return a JSON response with a 'response' field containing the reply: + + Email ID: {email.EmailId} + Content: {email.EmailContent} + """, + thread: emailThread); + + EmailResponse emailResponse = emailAssistantResponse.Result; + + return await context.CallActivityAsync(nameof(SendEmail), emailResponse.Response); + } + + [Function(nameof(HandleSpamEmail))] + public static string HandleSpamEmail([ActivityTrigger] string reason) + { + return $"Email marked as spam: {reason}"; + } + + [Function(nameof(SendEmail))] + public static string SendEmail([ActivityTrigger] string message) + { + return $"Email sent: {message}"; + } + + // POST /spamdetection/run + [Function(nameof(StartOrchestrationAsync))] + public static async Task StartOrchestrationAsync( + [HttpTrigger(AuthorizationLevel.Anonymous, "post", Route = "spamdetection/run")] HttpRequestData req, + [DurableClient] DurableTaskClient client) + { + // Read the email from the request body + Email? email = await req.ReadFromJsonAsync(); + if (email is null || string.IsNullOrWhiteSpace(email.EmailContent)) + { + HttpResponseData badRequestResponse = req.CreateResponse(HttpStatusCode.BadRequest); + await badRequestResponse.WriteAsJsonAsync(new { error = "Email with content is required" }); + return badRequestResponse; + } + + string instanceId = await client.ScheduleNewOrchestrationInstanceAsync( + orchestratorName: nameof(RunOrchestrationAsync), + input: email); + + HttpResponseData response = req.CreateResponse(HttpStatusCode.Accepted); + await response.WriteAsJsonAsync(new + { + message = "Spam detection orchestration started.", + emailId = email.EmailId, + instanceId, + statusQueryGetUri = GetStatusQueryGetUri(req, instanceId), + }); + return response; + } + + // GET /spamdetection/status/{instanceId} + [Function(nameof(GetOrchestrationStatusAsync))] + public static async Task GetOrchestrationStatusAsync( + [HttpTrigger(AuthorizationLevel.Anonymous, "get", Route = "spamdetection/status/{instanceId}")] HttpRequestData req, + string instanceId, + [DurableClient] DurableTaskClient client) + { + OrchestrationMetadata? status = await client.GetInstanceAsync( + instanceId, + getInputsAndOutputs: true, + req.FunctionContext.CancellationToken); + + if (status is null) + { + HttpResponseData notFound = req.CreateResponse(HttpStatusCode.NotFound); + await notFound.WriteAsJsonAsync(new { error = "Instance not found" }); + return notFound; + } + + HttpResponseData response = req.CreateResponse(HttpStatusCode.OK); + await response.WriteAsJsonAsync(new + { + instanceId = status.InstanceId, + runtimeStatus = status.RuntimeStatus.ToString(), + input = status.SerializedInput is not null ? (object)status.ReadInputAs() : null, + output = status.SerializedOutput is not null ? (object)status.ReadOutputAs() : null, + failureDetails = status.FailureDetails + }); + return response; + } + + private static string GetStatusQueryGetUri(HttpRequestData req, string instanceId) + { + // NOTE: This can be made more robust by considering the value of + // request headers like "X-Forwarded-Host" and "X-Forwarded-Proto". + string authority = $"{req.Url.Scheme}://{req.Url.Authority}"; + return $"{authority}/api/spamdetection/status/{instanceId}"; + } +} diff --git a/dotnet/samples/AzureFunctions/04_AgentOrchestration_Conditionals/Models.cs b/dotnet/samples/AzureFunctions/04_AgentOrchestration_Conditionals/Models.cs new file mode 100644 index 0000000000..a39695d7d0 --- /dev/null +++ b/dotnet/samples/AzureFunctions/04_AgentOrchestration_Conditionals/Models.cs @@ -0,0 +1,38 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +namespace AgentOrchestration_Conditionals; + +/// +/// Represents an email input for spam detection and response generation. +/// +public sealed class Email +{ + [JsonPropertyName("email_id")] + public string EmailId { get; set; } = string.Empty; + + [JsonPropertyName("email_content")] + public string EmailContent { get; set; } = string.Empty; +} + +/// +/// Represents the result of spam detection analysis. +/// +public sealed class DetectionResult +{ + [JsonPropertyName("is_spam")] + public bool IsSpam { get; set; } + + [JsonPropertyName("reason")] + public string Reason { get; set; } = string.Empty; +} + +/// +/// Represents a generated email response. +/// +public sealed class EmailResponse +{ + [JsonPropertyName("response")] + public string Response { get; set; } = string.Empty; +} diff --git a/dotnet/samples/AzureFunctions/04_AgentOrchestration_Conditionals/Program.cs b/dotnet/samples/AzureFunctions/04_AgentOrchestration_Conditionals/Program.cs new file mode 100644 index 0000000000..a04b4c3e70 --- /dev/null +++ b/dotnet/samples/AzureFunctions/04_AgentOrchestration_Conditionals/Program.cs @@ -0,0 +1,48 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Azure; +using Azure.AI.OpenAI; +using Azure.Identity; +using Microsoft.Agents.AI; +using Microsoft.Agents.AI.Hosting.AzureFunctions; +using Microsoft.Azure.Functions.Worker.Builder; +using Microsoft.Extensions.Hosting; +using OpenAI.Chat; + +// Get the Azure OpenAI endpoint and deployment name from environment variables. +string endpoint = Environment.GetEnvironmentVariable("AZURE_OPENAI_ENDPOINT") + ?? throw new InvalidOperationException("AZURE_OPENAI_ENDPOINT is not set."); +string deploymentName = Environment.GetEnvironmentVariable("AZURE_OPENAI_DEPLOYMENT") + ?? throw new InvalidOperationException("AZURE_OPENAI_DEPLOYMENT is not set."); + +// Use Azure Key Credential if provided, otherwise use Azure CLI Credential. +string? azureOpenAiKey = System.Environment.GetEnvironmentVariable("AZURE_OPENAI_KEY"); +AzureOpenAIClient client = !string.IsNullOrEmpty(azureOpenAiKey) + ? new AzureOpenAIClient(new Uri(endpoint), new AzureKeyCredential(azureOpenAiKey)) + : new AzureOpenAIClient(new Uri(endpoint), new AzureCliCredential()); + +// Two agents used by the orchestration to demonstrate conditional logic. +const string SpamDetectionName = "SpamDetectionAgent"; +const string SpamDetectionInstructions = "You are a spam detection assistant that identifies spam emails."; + +const string EmailAssistantName = "EmailAssistantAgent"; +const string EmailAssistantInstructions = "You are an email assistant that helps users draft responses to emails with professionalism."; + +AIAgent spamDetectionAgent = client.GetChatClient(deploymentName) + .CreateAIAgent(SpamDetectionInstructions, SpamDetectionName); + +AIAgent emailAssistantAgent = client.GetChatClient(deploymentName) + .CreateAIAgent(EmailAssistantInstructions, EmailAssistantName); + +using IHost app = FunctionsApplication + .CreateBuilder(args) + .ConfigureFunctionsWebApplication() + .ConfigureDurableAgents(options => + { + options + .AddAIAgent(spamDetectionAgent) + .AddAIAgent(emailAssistantAgent); + }) + .Build(); + +app.Run(); diff --git a/dotnet/samples/AzureFunctions/04_AgentOrchestration_Conditionals/README.md b/dotnet/samples/AzureFunctions/04_AgentOrchestration_Conditionals/README.md new file mode 100644 index 0000000000..97202b18a8 --- /dev/null +++ b/dotnet/samples/AzureFunctions/04_AgentOrchestration_Conditionals/README.md @@ -0,0 +1,113 @@ +# Multi-Agent Orchestration with Conditionals Sample + +This sample demonstrates how to use the Durable Agent Framework (DAFx) to create a multi-agent orchestration workflow that includes conditional logic. The workflow implements a spam detection system that processes emails and takes different actions based on whether the email is identified as spam or legitimate. + +## Key Concepts Demonstrated + +- Multi-agent orchestration with conditional logic and different processing paths +- Spam detection using AI agent analysis +- Structured output from agents for reliable processing +- Activity functions for integrating non-agentic workflow actions + +## Environment Setup + +See the [README.md](../README.md) file in the parent directory for more information on how to configure the environment, including how to install and run common sample dependencies. + +## Running the Sample + +With the environment setup and function app running, you can test the sample by sending an HTTP request with email data to the orchestration. + +You can use the `demo.http` file to send email data to the agents, or a command line tool like `curl` as shown below: + +Bash (Linux/macOS/WSL): + +```bash +# Test with a legitimate email +curl -X POST http://localhost:7071/api/spamdetection/run \ + -H "Content-Type: application/json" \ + -d '{ + "email_id": "email-001", + "email_content": "Hi John, I hope you are doing well. I wanted to follow up on our meeting yesterday about the quarterly report. Could you please send me the updated figures by Friday? Thanks!" + }' + +# Test with a spam email +curl -X POST http://localhost:7071/api/spamdetection/run \ + -H "Content-Type: application/json" \ + -d '{ + "email_id": "email-002", + "email_content": "URGENT! You have won $1,000,000! Click here now to claim your prize! Limited time offer! Do not miss out!" + }' +``` + +PowerShell: + +```powershell +# Test with a legitimate email +$body = @{ + email_id = "email-001" + email_content = "Hi John, I hope you are doing well. I wanted to follow up on our meeting yesterday about the quarterly report. Could you please send me the updated figures by Friday? Thanks!" +} | ConvertTo-Json + +Invoke-RestMethod -Method Post ` + -Uri http://localhost:7071/api/spamdetection/run ` + -ContentType application/json ` + -Body $body + +# Test with a spam email +$body = @{ + email_id = "email-002" + email_content = "URGENT! You have won $1,000,000! Click here now to claim your prize! Limited time offer! Do not miss out!" +} | ConvertTo-Json + +Invoke-RestMethod -Method Post ` + -Uri http://localhost:7071/api/spamdetection/run ` + -ContentType application/json ` + -Body $body +``` + +The response from either input will be a JSON object that looks something like the following, which indicates that the orchestration has started. + +```json +{ + "message": "Spam detection orchestration started.", + "emailId": "email-001", + "instanceId": "555dbbb63f75406db2edf9f1f092de95", + "statusQueryGetUri": "http://localhost:7071/api/spamdetection/status/555dbbb63f75406db2edf9f1f092de95" +} +``` + +The orchestration will: + +1. Analyze the email content using the SpamDetectionAgent +2. If spam: Mark the email as spam with a reason +3. If legitimate: Use the EmailAssistantAgent to draft a professional response and "send" it + +Once the orchestration has completed, you can get the status of the orchestration by sending a GET request to the `statusQueryGetUri` URL. The response for the legitimate email will be a JSON object that looks something like the following: + +```json +{ + "failureDetails": null, + "input": { + "email_content": "Hi John, I hope you're doing well. I wanted to follow up on our meeting yesterday about the quarterly report. Could you please send me the updated figures by Friday? Thanks!", + "email_id": "email-001" + }, + "instanceId": "555dbbb63f75406db2edf9f1f092de95", + "output": "Email sent: Subject: Re: Follow-Up on Quarterly Report\n\nHi [Recipient's Name],\n\nI hope this message finds you well. Thank you for your patience. I will ensure the updated figures for the quarterly report are sent to you by Friday.\n\nIf you have any further questions or need additional information, please feel free to reach out.\n\nBest regards,\n\nJohn", + "runtimeStatus": "Completed" +} +``` + +The response for the spam email will be a JSON object that looks something like the following, which indicates that the email was marked as spam: + +```json +{ + "failureDetails": null, + "input": { + "email_content": "URGENT! You have won $1,000,000! Click here now to claim your prize! Limited time offer! Do not miss out!", + "email_id": "email-002" + }, + "instanceId": "555dbbb63f75406db2edf9f1f092de95", + "output": "Email marked as spam: The email contains misleading claims of winning a large sum of money and encourages immediate action, which are common characteristics of spam.", + "runtimeStatus": "Completed" +} +``` diff --git a/dotnet/samples/AzureFunctions/04_AgentOrchestration_Conditionals/demo.http b/dotnet/samples/AzureFunctions/04_AgentOrchestration_Conditionals/demo.http new file mode 100644 index 0000000000..1120a7a181 --- /dev/null +++ b/dotnet/samples/AzureFunctions/04_AgentOrchestration_Conditionals/demo.http @@ -0,0 +1,18 @@ +### Test spam detection with a legitimate email +POST http://localhost:7071/api/spamdetection/run +Content-Type: application/json + +{ + "email_id": "email-001", + "email_content": "Hi John, I hope you're doing well. I wanted to follow up on our meeting yesterday about the quarterly report. Could you please send me the updated figures by Friday? Thanks!" +} + + +### Test spam detection with a spam email +POST http://localhost:7071/api/spamdetection/run +Content-Type: application/json + +{ + "email_id": "email-002", + "email_content": "URGENT! You've won $1,000,000! Click here now to claim your prize! Limited time offer! Don't miss out!" +} diff --git a/dotnet/samples/AzureFunctions/04_AgentOrchestration_Conditionals/host.json b/dotnet/samples/AzureFunctions/04_AgentOrchestration_Conditionals/host.json new file mode 100644 index 0000000000..9384a0a583 --- /dev/null +++ b/dotnet/samples/AzureFunctions/04_AgentOrchestration_Conditionals/host.json @@ -0,0 +1,20 @@ +{ + "version": "2.0", + "logging": { + "logLevel": { + "Microsoft.Agents.AI.DurableTask": "Information", + "Microsoft.Agents.AI.Hosting.AzureFunctions": "Information", + "DurableTask": "Information", + "Microsoft.DurableTask": "Information" + } + }, + "extensions": { + "durableTask": { + "hubName": "default", + "storageProvider": { + "type": "AzureManaged", + "connectionStringName": "DURABLE_TASK_SCHEDULER_CONNECTION_STRING" + } + } + } +} diff --git a/dotnet/samples/AzureFunctions/04_AgentOrchestration_Conditionals/local.settings.json b/dotnet/samples/AzureFunctions/04_AgentOrchestration_Conditionals/local.settings.json new file mode 100644 index 0000000000..54dfbb5664 --- /dev/null +++ b/dotnet/samples/AzureFunctions/04_AgentOrchestration_Conditionals/local.settings.json @@ -0,0 +1,10 @@ +{ + "IsEncrypted": false, + "Values": { + "FUNCTIONS_WORKER_RUNTIME": "dotnet-isolated", + "AzureWebJobsStorage": "UseDevelopmentStorage=true", + "DURABLE_TASK_SCHEDULER_CONNECTION_STRING": "Endpoint=http://localhost:8080;TaskHub=default;Authentication=None", + "AZURE_OPENAI_ENDPOINT": "", + "AZURE_OPENAI_DEPLOYMENT": "" + } +} diff --git a/dotnet/samples/AzureFunctions/05_AgentOrchestration_HITL/05_AgentOrchestration_HITL.csproj b/dotnet/samples/AzureFunctions/05_AgentOrchestration_HITL/05_AgentOrchestration_HITL.csproj new file mode 100644 index 0000000000..a240ea0394 --- /dev/null +++ b/dotnet/samples/AzureFunctions/05_AgentOrchestration_HITL/05_AgentOrchestration_HITL.csproj @@ -0,0 +1,43 @@ + + + net10.0 + v4 + Exe + enable + enable + + AgentOrchestration_HITL + AgentOrchestration_HITL + $(NoWarn);DURABLE0001;DURABLE0002;DURABLE0003;DURABLE0004;DURABLE0005;DURABLE0006 + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/dotnet/samples/AzureFunctions/05_AgentOrchestration_HITL/FunctionTriggers.cs b/dotnet/samples/AzureFunctions/05_AgentOrchestration_HITL/FunctionTriggers.cs new file mode 100644 index 0000000000..001a52c105 --- /dev/null +++ b/dotnet/samples/AzureFunctions/05_AgentOrchestration_HITL/FunctionTriggers.cs @@ -0,0 +1,229 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Net; +using System.Text.Json; +using Microsoft.Agents.AI; +using Microsoft.Agents.AI.DurableTask; +using Microsoft.Azure.Functions.Worker; +using Microsoft.Azure.Functions.Worker.Http; +using Microsoft.DurableTask; +using Microsoft.DurableTask.Client; +using Microsoft.Extensions.Logging; + +namespace AgentOrchestration_HITL; + +public static class FunctionTriggers +{ + [Function(nameof(RunOrchestrationAsync))] + public static async Task RunOrchestrationAsync( + [OrchestrationTrigger] TaskOrchestrationContext context) + { + // Get the input from the orchestration + ContentGenerationInput input = context.GetInput() + ?? throw new InvalidOperationException("Content generation input is required"); + + // Get the writer agent + DurableAIAgent writerAgent = context.GetAgent("WriterAgent"); + AgentThread writerThread = writerAgent.GetNewThread(); + + // Set initial status + context.SetCustomStatus($"Starting content generation for topic: {input.Topic}"); + + // Step 1: Generate initial content + AgentRunResponse writerResponse = await writerAgent.RunAsync( + message: $"Write a short article about '{input.Topic}'.", + thread: writerThread); + GeneratedContent content = writerResponse.Result; + + // Human-in-the-loop iteration - we set a maximum number of attempts to avoid infinite loops + int iterationCount = 0; + while (iterationCount++ < input.MaxReviewAttempts) + { + context.SetCustomStatus( + $"Requesting human feedback. Iteration #{iterationCount}. Timeout: {input.ApprovalTimeoutHours} hour(s)."); + + // Step 2: Notify user to review the content + await context.CallActivityAsync(nameof(NotifyUserForApproval), content); + + // Step 3: Wait for human feedback with configurable timeout + HumanApprovalResponse humanResponse; + try + { + humanResponse = await context.WaitForExternalEvent( + eventName: "HumanApproval", + timeout: TimeSpan.FromHours(input.ApprovalTimeoutHours)); + } + catch (OperationCanceledException) + { + // Timeout occurred - treat as rejection + context.SetCustomStatus( + $"Human approval timed out after {input.ApprovalTimeoutHours} hour(s). Treating as rejection."); + throw new TimeoutException($"Human approval timed out after {input.ApprovalTimeoutHours} hour(s)."); + } + + if (humanResponse.Approved) + { + context.SetCustomStatus("Content approved by human reviewer. Publishing content..."); + + // Step 4: Publish the approved content + await context.CallActivityAsync(nameof(PublishContent), content); + + context.SetCustomStatus($"Content published successfully at {context.CurrentUtcDateTime:s}"); + return new { content = content.Content }; + } + + context.SetCustomStatus("Content rejected by human reviewer. Incorporating feedback and regenerating..."); + + // Incorporate human feedback and regenerate + writerResponse = await writerAgent.RunAsync( + message: $""" + The content was rejected by a human reviewer. Please rewrite the article incorporating their feedback. + + Human Feedback: {humanResponse.Feedback} + """, + thread: writerThread); + + content = writerResponse.Result; + } + + // If we reach here, it means we exhausted the maximum number of iterations + throw new InvalidOperationException( + $"Content could not be approved after {input.MaxReviewAttempts} iterations."); + } + + // POST /hitl/run + [Function(nameof(StartOrchestrationAsync))] + public static async Task StartOrchestrationAsync( + [HttpTrigger(AuthorizationLevel.Anonymous, "post", Route = "hitl/run")] HttpRequestData req, + [DurableClient] DurableTaskClient client) + { + // Read the input from the request body + ContentGenerationInput? input = await req.ReadFromJsonAsync(); + if (input is null || string.IsNullOrWhiteSpace(input.Topic)) + { + HttpResponseData badRequestResponse = req.CreateResponse(HttpStatusCode.BadRequest); + await badRequestResponse.WriteAsJsonAsync(new { error = "Topic is required" }); + return badRequestResponse; + } + + string instanceId = await client.ScheduleNewOrchestrationInstanceAsync( + orchestratorName: nameof(RunOrchestrationAsync), + input: input); + + HttpResponseData response = req.CreateResponse(HttpStatusCode.Accepted); + await response.WriteAsJsonAsync(new + { + message = "HITL content generation orchestration started.", + topic = input.Topic, + instanceId, + statusQueryGetUri = GetStatusQueryGetUri(req, instanceId), + }); + return response; + } + + // POST /hitl/approve/{instanceId} + [Function(nameof(SendHumanApprovalAsync))] + public static async Task SendHumanApprovalAsync( + [HttpTrigger(AuthorizationLevel.Anonymous, "post", Route = "hitl/approve/{instanceId}")] HttpRequestData req, + string instanceId, + [DurableClient] DurableTaskClient client) + { + // Read the approval response from the request body + HumanApprovalResponse? approvalResponse = await req.ReadFromJsonAsync(); + if (approvalResponse is null) + { + HttpResponseData badRequestResponse = req.CreateResponse(HttpStatusCode.BadRequest); + await badRequestResponse.WriteAsJsonAsync(new { error = "Approval response is required" }); + return badRequestResponse; + } + + // Send the approval event to the orchestration + await client.RaiseEventAsync(instanceId, "HumanApproval", approvalResponse); + + HttpResponseData response = req.CreateResponse(HttpStatusCode.OK); + await response.WriteAsJsonAsync(new + { + message = "Human approval sent to orchestration.", + instanceId, + approved = approvalResponse.Approved + }); + return response; + } + + // GET /hitl/status/{instanceId} + [Function(nameof(GetOrchestrationStatusAsync))] + public static async Task GetOrchestrationStatusAsync( + [HttpTrigger(AuthorizationLevel.Anonymous, "get", Route = "hitl/status/{instanceId}")] HttpRequestData req, + string instanceId, + [DurableClient] DurableTaskClient client) + { + OrchestrationMetadata? status = await client.GetInstanceAsync( + instanceId, + getInputsAndOutputs: true, + req.FunctionContext.CancellationToken); + + if (status is null) + { + HttpResponseData notFound = req.CreateResponse(HttpStatusCode.NotFound); + await notFound.WriteAsJsonAsync(new { error = "Instance not found" }); + return notFound; + } + + HttpResponseData response = req.CreateResponse(HttpStatusCode.OK); + await response.WriteAsJsonAsync(new + { + instanceId = status.InstanceId, + runtimeStatus = status.RuntimeStatus.ToString(), + workflowStatus = status.SerializedCustomStatus is not null ? (object)status.ReadCustomStatusAs() : null, + input = status.SerializedInput is not null ? (object)status.ReadInputAs() : null, + output = status.SerializedOutput is not null ? (object)status.ReadOutputAs() : null, + failureDetails = status.FailureDetails + }); + return response; + } + + [Function(nameof(NotifyUserForApproval))] + public static void NotifyUserForApproval( + [ActivityTrigger] GeneratedContent content, + FunctionContext functionContext) + { + ILogger logger = functionContext.GetLogger(nameof(NotifyUserForApproval)); + + // In a real implementation, this would send notifications via email, SMS, etc. + logger.LogInformation( + """ + NOTIFICATION: Please review the following content for approval: + Title: {Title} + Content: {Content} + Use the approval endpoint to approve or reject this content. + """, + content.Title, + content.Content); + } + + [Function(nameof(PublishContent))] + public static void PublishContent( + [ActivityTrigger] GeneratedContent content, + FunctionContext functionContext) + { + ILogger logger = functionContext.GetLogger(nameof(PublishContent)); + + // In a real implementation, this would publish to a CMS, website, etc. + logger.LogInformation( + """ + PUBLISHING: Content has been published successfully. + Title: {Title} + Content: {Content} + """, + content.Title, + content.Content); + } + + private static string GetStatusQueryGetUri(HttpRequestData req, string instanceId) + { + // NOTE: This can be made more robust by considering the value of + // request headers like "X-Forwarded-Host" and "X-Forwarded-Proto". + string authority = $"{req.Url.Scheme}://{req.Url.Authority}"; + return $"{authority}/api/hitl/status/{instanceId}"; + } +} diff --git a/dotnet/samples/AzureFunctions/05_AgentOrchestration_HITL/Models.cs b/dotnet/samples/AzureFunctions/05_AgentOrchestration_HITL/Models.cs new file mode 100644 index 0000000000..1eaf1407eb --- /dev/null +++ b/dotnet/samples/AzureFunctions/05_AgentOrchestration_HITL/Models.cs @@ -0,0 +1,44 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +namespace AgentOrchestration_HITL; + +/// +/// Represents the input for the Human-in-the-Loop content generation workflow. +/// +public sealed class ContentGenerationInput +{ + [JsonPropertyName("topic")] + public string Topic { get; set; } = string.Empty; + + [JsonPropertyName("max_review_attempts")] + public int MaxReviewAttempts { get; set; } = 3; + + [JsonPropertyName("approval_timeout_hours")] + public float ApprovalTimeoutHours { get; set; } = 72; +} + +/// +/// Represents the content generated by the writer agent. +/// +public sealed class GeneratedContent +{ + [JsonPropertyName("title")] + public string Title { get; set; } = string.Empty; + + [JsonPropertyName("content")] + public string Content { get; set; } = string.Empty; +} + +/// +/// Represents the human approval response. +/// +public sealed class HumanApprovalResponse +{ + [JsonPropertyName("approved")] + public bool Approved { get; set; } + + [JsonPropertyName("feedback")] + public string Feedback { get; set; } = string.Empty; +} diff --git a/dotnet/samples/AzureFunctions/05_AgentOrchestration_HITL/Program.cs b/dotnet/samples/AzureFunctions/05_AgentOrchestration_HITL/Program.cs new file mode 100644 index 0000000000..741e5407e0 --- /dev/null +++ b/dotnet/samples/AzureFunctions/05_AgentOrchestration_HITL/Program.cs @@ -0,0 +1,40 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Azure; +using Azure.AI.OpenAI; +using Azure.Identity; +using Microsoft.Agents.AI; +using Microsoft.Agents.AI.Hosting.AzureFunctions; +using Microsoft.Azure.Functions.Worker.Builder; +using Microsoft.Extensions.Hosting; +using OpenAI.Chat; + +// Get the Azure OpenAI endpoint and deployment name from environment variables. +string endpoint = Environment.GetEnvironmentVariable("AZURE_OPENAI_ENDPOINT") + ?? throw new InvalidOperationException("AZURE_OPENAI_ENDPOINT is not set."); +string deploymentName = Environment.GetEnvironmentVariable("AZURE_OPENAI_DEPLOYMENT") + ?? throw new InvalidOperationException("AZURE_OPENAI_DEPLOYMENT is not set."); + +// Use Azure Key Credential if provided, otherwise use Azure CLI Credential. +string? azureOpenAiKey = System.Environment.GetEnvironmentVariable("AZURE_OPENAI_KEY"); +AzureOpenAIClient client = !string.IsNullOrEmpty(azureOpenAiKey) + ? new AzureOpenAIClient(new Uri(endpoint), new AzureKeyCredential(azureOpenAiKey)) + : new AzureOpenAIClient(new Uri(endpoint), new AzureCliCredential()); + +// Single agent used by the orchestration to demonstrate human-in-the-loop workflow. +const string WriterName = "WriterAgent"; +const string WriterInstructions = + """ + You are a professional content writer who creates high-quality articles on various topics. + You write engaging, informative, and well-structured content that follows best practices for readability and accuracy. + """; + +AIAgent writerAgent = client.GetChatClient(deploymentName).CreateAIAgent(WriterInstructions, WriterName); + +using IHost app = FunctionsApplication + .CreateBuilder(args) + .ConfigureFunctionsWebApplication() + .ConfigureDurableAgents(options => options.AddAIAgent(writerAgent)) + .Build(); + +app.Run(); diff --git a/dotnet/samples/AzureFunctions/05_AgentOrchestration_HITL/README.md b/dotnet/samples/AzureFunctions/05_AgentOrchestration_HITL/README.md new file mode 100644 index 0000000000..b6aa2f037a --- /dev/null +++ b/dotnet/samples/AzureFunctions/05_AgentOrchestration_HITL/README.md @@ -0,0 +1,126 @@ +# Multi-Agent Orchestration with Human-in-the-Loop Sample + +This sample demonstrates how to use the Durable Agent Framework (DAFx) to create a human-in-the-loop (HITL) workflow using a single AI agent. The workflow uses a writer agent to generate content and requires human approval on every iteration, emphasizing the human-in-the-loop pattern. + +## Key Concepts Demonstrated + +- Single-agent orchestration +- Human-in-the-loop feedback loop using external events (`WaitForExternalEvent`) +- Activity functions for non-agentic workflow steps +- Iterative content refinement based on human feedback +- Custom status tracking for workflow visibility +- Error handling with maximum retry attempts and timeout handling for human approval + +## Environment Setup + +See the [README.md](../README.md) file in the parent directory for more information on how to configure the environment, including how to install and run common sample dependencies. + +## Running the Sample + +With the environment setup and function app running, you can test the sample by sending an HTTP request with a topic to start the content generation workflow. + +You can use the `demo.http` file to send a topic to the agents, or a command line tool like `curl` as shown below: + +Bash (Linux/macOS/WSL): + +```bash +curl -X POST http://localhost:7071/api/hitl/run \ + -H "Content-Type: application/json" \ + -d '{ + "topic": "The Future of Artificial Intelligence", + "max_review_attempts": 3, + "timeout_minutes": 5 + }' +``` + +PowerShell: + +```powershell +$body = @{ + topic = "The Future of Artificial Intelligence" + max_review_attempts = 3 + timeout_minutes = 5 +} | ConvertTo-Json + +Invoke-RestMethod -Method Post ` + -Uri http://localhost:7071/api/hitl/run ` + -ContentType application/json ` + -Body $body +``` + +The response will be a JSON object that looks something like the following, which indicates that the orchestration has started. + +```json +{ + "message": "HITL content generation orchestration started.", + "topic": "The Future of Artificial Intelligence", + "instanceId": "a1b2c3d4e5f6g7h8i9j0k1l2m3n4o5p6", + "statusQueryGetUri": "http://localhost:7071/api/hitl/status/a1b2c3d4e5f6g7h8i9j0k1l2m3n4o5p6" +} +``` + +The orchestration will: + +1. Generate initial content using the WriterAgent +2. Notify the user to review the content +3. Wait for human feedback via external event (configurable timeout) +4. If approved by human, publish the content +5. If rejected by human, incorporate feedback and regenerate content +6. If approval timeout occurs, treat as rejection and fail the orchestration +7. Repeat until human approval is received or maximum loop iterations are reached + +Once the orchestration is waiting for human approval, you can send approval or rejection using the approval endpoint: + +Bash (Linux/macOS/WSL): + +```bash +# Approve the content +curl -X POST http://localhost:7071/api/hitl/approve/a1b2c3d4e5f6g7h8i9j0k1l2m3n4o5p6 \ + -H "Content-Type: application/json" \ + -d '{ + "approved": true, + "feedback": "Great article! The content is well-structured and informative." + }' + +# Reject the content with feedback +curl -X POST http://localhost:7071/api/hitl/approve/a1b2c3d4e5f6g7h8i9j0k1l2m3n4o5p6 \ + -H "Content-Type: application/json" \ + -d '{ + "approved": false, + "feedback": "The article needs more technical depth and better examples." + }' +``` + +PowerShell: + +```powershell +# Approve the content +Invoke-RestMethod -Method Post ` + -Uri http://localhost:7071/api/hitl/approve/a1b2c3d4e5f6g7h8i9j0k1l2m3n4o5p6 ` + -ContentType application/json ` + -Body '{ "approved": true, "feedback": "Great article! The content is well-structured and informative." }' + +# Reject the content with feedback +Invoke-RestMethod -Method Post ` + -Uri http://localhost:7071/api/hitl/approve/a1b2c3d4e5f6g7h8i9j0k1l2m3n4o5p6 ` + -ContentType application/json ` + -Body '{ "approved": false, "feedback": "The article needs more technical depth and better examples." }' +``` + +Once the orchestration has completed, you can get the status by sending a GET request to the `statusQueryGetUri` URL. The response will be a JSON object that looks something like the following: + +```json +{ + "failureDetails": null, + "input": { + "topic": "The Future of Artificial Intelligence", + "max_review_attempts": 3 + }, + "instanceId": "a1b2c3d4e5f6g7h8i9j0k1l2m3n4o5p6", + "output": { + "content": "The Future of Artificial Intelligence is..." + }, + "runtimeStatus": "Completed", + "workflowStatus": "Content published successfully at 2025-10-15T12:00:00Z" +} +``` diff --git a/dotnet/samples/AzureFunctions/05_AgentOrchestration_HITL/demo.http b/dotnet/samples/AzureFunctions/05_AgentOrchestration_HITL/demo.http new file mode 100644 index 0000000000..2ab2dc428a --- /dev/null +++ b/dotnet/samples/AzureFunctions/05_AgentOrchestration_HITL/demo.http @@ -0,0 +1,44 @@ +### Start the HITL content generation orchestration with default timeout (30 days) +POST http://localhost:7071/api/hitl/run +Content-Type: application/json + +{ + "topic": "The Future of Artificial Intelligence", + "max_review_attempts": 3 +} + + +### Start the HITL content generation orchestration with very short timeout for demonstration (~4 seconds) +POST http://localhost:7071/api/hitl/run +Content-Type: application/json + +{ + "topic": "The Future of Artificial Intelligence", + "max_review_attempts": 3, + "approval_timeout_hours": 0.001 +} + + +### Copy/paste the instanceId from the response above +@instanceId=INSTANCE_ID_GOES_HERE + +### Check the status of the orchestration (replace {instanceId} with the actual instance ID from the response above) +GET http://localhost:7071/api/hitl/status/{{instanceId}} + +### Send human approval (replace {instanceId} with the actual instance ID) +POST http://localhost:7071/api/hitl/approve/{{instanceId}} +Content-Type: application/json + +{ + "approved": true, + "feedback": "Great article! The content is well-structured and informative." +} + +### Send human rejection with feedback (replace {instanceId} with the actual instance ID) +POST http://localhost:7071/api/hitl/approve/{{instanceId}} +Content-Type: application/json + +{ + "approved": false, + "feedback": "The article needs more technical depth and better examples. Please add more specific use cases and implementation details." +} diff --git a/dotnet/samples/AzureFunctions/05_AgentOrchestration_HITL/host.json b/dotnet/samples/AzureFunctions/05_AgentOrchestration_HITL/host.json new file mode 100644 index 0000000000..9384a0a583 --- /dev/null +++ b/dotnet/samples/AzureFunctions/05_AgentOrchestration_HITL/host.json @@ -0,0 +1,20 @@ +{ + "version": "2.0", + "logging": { + "logLevel": { + "Microsoft.Agents.AI.DurableTask": "Information", + "Microsoft.Agents.AI.Hosting.AzureFunctions": "Information", + "DurableTask": "Information", + "Microsoft.DurableTask": "Information" + } + }, + "extensions": { + "durableTask": { + "hubName": "default", + "storageProvider": { + "type": "AzureManaged", + "connectionStringName": "DURABLE_TASK_SCHEDULER_CONNECTION_STRING" + } + } + } +} diff --git a/dotnet/samples/AzureFunctions/05_AgentOrchestration_HITL/local.settings.json b/dotnet/samples/AzureFunctions/05_AgentOrchestration_HITL/local.settings.json new file mode 100644 index 0000000000..54dfbb5664 --- /dev/null +++ b/dotnet/samples/AzureFunctions/05_AgentOrchestration_HITL/local.settings.json @@ -0,0 +1,10 @@ +{ + "IsEncrypted": false, + "Values": { + "FUNCTIONS_WORKER_RUNTIME": "dotnet-isolated", + "AzureWebJobsStorage": "UseDevelopmentStorage=true", + "DURABLE_TASK_SCHEDULER_CONNECTION_STRING": "Endpoint=http://localhost:8080;TaskHub=default;Authentication=None", + "AZURE_OPENAI_ENDPOINT": "", + "AZURE_OPENAI_DEPLOYMENT": "" + } +} diff --git a/dotnet/samples/AzureFunctions/06_LongRunningTools/06_LongRunningTools.csproj b/dotnet/samples/AzureFunctions/06_LongRunningTools/06_LongRunningTools.csproj new file mode 100644 index 0000000000..8711331aa2 --- /dev/null +++ b/dotnet/samples/AzureFunctions/06_LongRunningTools/06_LongRunningTools.csproj @@ -0,0 +1,42 @@ + + + net10.0 + v4 + Exe + enable + enable + + LongRunningTools + LongRunningTools + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/dotnet/samples/AzureFunctions/06_LongRunningTools/FunctionTriggers.cs b/dotnet/samples/AzureFunctions/06_LongRunningTools/FunctionTriggers.cs new file mode 100644 index 0000000000..b5f81276b8 --- /dev/null +++ b/dotnet/samples/AzureFunctions/06_LongRunningTools/FunctionTriggers.cs @@ -0,0 +1,151 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Microsoft.Agents.AI; +using Microsoft.Agents.AI.DurableTask; +using Microsoft.Azure.Functions.Worker; +using Microsoft.DurableTask; +using Microsoft.Extensions.Logging; + +namespace LongRunningTools; + +public static class FunctionTriggers +{ + [Function(nameof(RunOrchestrationAsync))] + public static async Task RunOrchestrationAsync( + [OrchestrationTrigger] TaskOrchestrationContext context) + { + // Get the input from the orchestration + ContentGenerationInput input = context.GetInput() + ?? throw new InvalidOperationException("Content generation input is required"); + + // Get the writer agent + DurableAIAgent writerAgent = context.GetAgent("Writer"); + AgentThread writerThread = writerAgent.GetNewThread(); + + // Set initial status + context.SetCustomStatus($"Starting content generation for topic: {input.Topic}"); + + // Step 1: Generate initial content + AgentRunResponse writerResponse = await writerAgent.RunAsync( + message: $"Write a short article about '{input.Topic}'.", + thread: writerThread); + GeneratedContent content = writerResponse.Result; + + // Human-in-the-loop iteration - we set a maximum number of attempts to avoid infinite loops + int iterationCount = 0; + while (iterationCount++ < input.MaxReviewAttempts) + { + context.SetCustomStatus( + new + { + message = "Requesting human feedback.", + approvalTimeoutHours = input.ApprovalTimeoutHours, + iterationCount, + content + }); + + // Step 2: Notify user to review the content + await context.CallActivityAsync(nameof(NotifyUserForApproval), content); + + // Step 3: Wait for human feedback with configurable timeout + HumanApprovalResponse humanResponse; + try + { + humanResponse = await context.WaitForExternalEvent( + eventName: "HumanApproval", + timeout: TimeSpan.FromHours(input.ApprovalTimeoutHours)); + } + catch (OperationCanceledException) + { + // Timeout occurred - treat as rejection + context.SetCustomStatus( + new + { + message = $"Human approval timed out after {input.ApprovalTimeoutHours} hour(s). Treating as rejection.", + iterationCount, + content + }); + throw new TimeoutException($"Human approval timed out after {input.ApprovalTimeoutHours} hour(s)."); + } + + if (humanResponse.Approved) + { + context.SetCustomStatus(new + { + message = "Content approved by human reviewer. Publishing content...", + content + }); + + // Step 4: Publish the approved content + await context.CallActivityAsync(nameof(PublishContent), content); + + context.SetCustomStatus(new + { + message = $"Content published successfully at {context.CurrentUtcDateTime:s}", + humanFeedback = humanResponse, + content + }); + return new { content = content.Content }; + } + + context.SetCustomStatus(new + { + message = "Content rejected by human reviewer. Incorporating feedback and regenerating...", + humanFeedback = humanResponse, + content + }); + + // Incorporate human feedback and regenerate + writerResponse = await writerAgent.RunAsync( + message: $""" + The content was rejected by a human reviewer. Please rewrite the article incorporating their feedback. + + Human Feedback: {humanResponse.Feedback} + """, + thread: writerThread); + + content = writerResponse.Result; + } + + // If we reach here, it means we exhausted the maximum number of iterations + throw new InvalidOperationException( + $"Content could not be approved after {input.MaxReviewAttempts} iterations."); + } + + [Function(nameof(NotifyUserForApproval))] + public static void NotifyUserForApproval( + [ActivityTrigger] GeneratedContent content, + FunctionContext functionContext) + { + ILogger logger = functionContext.GetLogger(nameof(NotifyUserForApproval)); + + // In a real implementation, this would send notifications via email, SMS, etc. + logger.LogInformation( + """ + NOTIFICATION: Please review the following content for approval: + Title: {Title} + Content: {Content} + Use the approval endpoint to approve or reject this content. + """, + content.Title, + content.Content); + } + + [Function(nameof(PublishContent))] + public static void PublishContent( + [ActivityTrigger] GeneratedContent content, + FunctionContext functionContext) + { + ILogger logger = functionContext.GetLogger(nameof(PublishContent)); + + // In a real implementation, this would publish to a CMS, website, etc. + logger.LogInformation( + """ + PUBLISHING: Content has been published successfully. + Title: {Title} + Content: {Content} + """, + content.Title, + content.Content); + } +} diff --git a/dotnet/samples/AzureFunctions/06_LongRunningTools/Models.cs b/dotnet/samples/AzureFunctions/06_LongRunningTools/Models.cs new file mode 100644 index 0000000000..771343694d --- /dev/null +++ b/dotnet/samples/AzureFunctions/06_LongRunningTools/Models.cs @@ -0,0 +1,44 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +namespace LongRunningTools; + +/// +/// Represents the input for the content generation workflow. +/// +public sealed class ContentGenerationInput +{ + [JsonPropertyName("topic")] + public string Topic { get; set; } = string.Empty; + + [JsonPropertyName("max_review_attempts")] + public int MaxReviewAttempts { get; set; } = 3; + + [JsonPropertyName("approval_timeout_hours")] + public float ApprovalTimeoutHours { get; set; } = 72; +} + +/// +/// Represents the content generated by the writer agent. +/// +public sealed class GeneratedContent +{ + [JsonPropertyName("title")] + public string Title { get; set; } = string.Empty; + + [JsonPropertyName("content")] + public string Content { get; set; } = string.Empty; +} + +/// +/// Represents the human approval response. +/// +public sealed class HumanApprovalResponse +{ + [JsonPropertyName("approved")] + public bool Approved { get; set; } + + [JsonPropertyName("feedback")] + public string Feedback { get; set; } = string.Empty; +} diff --git a/dotnet/samples/AzureFunctions/06_LongRunningTools/Program.cs b/dotnet/samples/AzureFunctions/06_LongRunningTools/Program.cs new file mode 100644 index 0000000000..581b2bce11 --- /dev/null +++ b/dotnet/samples/AzureFunctions/06_LongRunningTools/Program.cs @@ -0,0 +1,73 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Azure; +using Azure.AI.OpenAI; +using Azure.Identity; +using LongRunningTools; +using Microsoft.Agents.AI; +using Microsoft.Agents.AI.Hosting.AzureFunctions; +using Microsoft.Azure.Functions.Worker.Builder; +using Microsoft.Extensions.AI; +using Microsoft.Extensions.DependencyInjection; +using Microsoft.Extensions.Hosting; +using Microsoft.Extensions.Logging; +using OpenAI.Chat; + +// Get the Azure OpenAI endpoint and deployment name from environment variables. +string endpoint = Environment.GetEnvironmentVariable("AZURE_OPENAI_ENDPOINT") + ?? throw new InvalidOperationException("AZURE_OPENAI_ENDPOINT is not set."); +string deploymentName = Environment.GetEnvironmentVariable("AZURE_OPENAI_DEPLOYMENT") + ?? throw new InvalidOperationException("AZURE_OPENAI_DEPLOYMENT is not set."); + +// Use Azure Key Credential if provided, otherwise use Azure CLI Credential. +string? azureOpenAiKey = System.Environment.GetEnvironmentVariable("AZURE_OPENAI_KEY"); +AzureOpenAIClient client = !string.IsNullOrEmpty(azureOpenAiKey) + ? new AzureOpenAIClient(new Uri(endpoint), new AzureKeyCredential(azureOpenAiKey)) + : new AzureOpenAIClient(new Uri(endpoint), new AzureCliCredential()); + +// Agent used by the orchestration to write content. +const string WriterAgentName = "Writer"; +const string WriterAgentInstructions = + """ + You are a professional content writer who creates high-quality articles on various topics. + You write engaging, informative, and well-structured content that follows best practices for readability and accuracy. + """; + +AIAgent writerAgent = client.GetChatClient(deploymentName).CreateAIAgent(WriterAgentInstructions, WriterAgentName); + +// Agent that can start content generation workflows using tools +const string PublisherAgentName = "Publisher"; +const string PublisherAgentInstructions = + """ + You are a publishing agent that can manage content generation workflows. + You have access to tools to start, monitor, and raise events for content generation workflows. + """; + +using IHost app = FunctionsApplication + .CreateBuilder(args) + .ConfigureFunctionsWebApplication() + .ConfigureDurableAgents(options => + { + // Add the writer agent used by the orchestration + options.AddAIAgent(writerAgent); + + // Define the agent that can start orchestrations from tool calls + options.AddAIAgentFactory(PublisherAgentName, sp => + { + // Initialize the tools to be used by the agent. + Tools publisherTools = new(sp.GetRequiredService>()); + + return client.GetChatClient(deploymentName).CreateAIAgent( + instructions: PublisherAgentInstructions, + name: PublisherAgentName, + services: sp, + tools: [ + AIFunctionFactory.Create(publisherTools.StartContentGenerationWorkflow), + AIFunctionFactory.Create(publisherTools.GetWorkflowStatusAsync), + AIFunctionFactory.Create(publisherTools.SubmitHumanApprovalAsync), + ]); + }); + }) + .Build(); + +app.Run(); diff --git a/dotnet/samples/AzureFunctions/06_LongRunningTools/README.md b/dotnet/samples/AzureFunctions/06_LongRunningTools/README.md new file mode 100644 index 0000000000..54ed85060b --- /dev/null +++ b/dotnet/samples/AzureFunctions/06_LongRunningTools/README.md @@ -0,0 +1,129 @@ +# Long Running Tools Sample + +This sample demonstrates how to use the Durable Agent Framework (DAFx) to create agents with long running tools. This sample builds on the [05_AgentOrchestration_HITL](../05_AgentOrchestration_HITL) sample by adding a publisher agent that can start and manage content generation workflows. A key difference is that the publisher agent knows the IDs of the workflows it starts, so it can check the status of the workflows and approve or reject them without being explicitly given the context (instance IDs, etc). + +## Key Concepts Demonstrated + +The same key concepts as the [05_AgentOrchestration_HITL](../05_AgentOrchestration_HITL) sample are demonstrated, but with the following additional concepts: + +- **Long running tools**: Using `DurableAgentContext.Current` to start orchestrations from tool calls +- **Multi-agent orchestration**: Agents can start and manage workflows that orchestrate other agents +- **Human-in-the-loop (with delegation)**: The agent acts as an intermediary between the human and the workflow. The human remains in the loop, but delegates to the agent to start the workflow and approve or reject the content. + +## Environment Setup + +See the [README.md](../README.md) file in the parent directory for more information on how to configure the environment, including how to install and run common sample dependencies. + +## Running the Sample + +With the environment setup and function app running, you can test the sample by sending an HTTP request to start the agent, which will then trigger the content generation workflow. + +You can use the `demo.http` file to send requests to the agent, or a command line tool like `curl` as shown below. + +Bash (Linux/macOS/WSL): + +```bash +curl -i -X POST http://localhost:7071/api/agents/publisher/run \ + -D headers.txt \ + -H "Content-Type: text/plain" \ + -d 'Start a content generation workflow for the topic \"The Future of Artificial Intelligence\"' + +# Save the thread ID to a variable and print it to the terminal +threadId=$(cat headers.txt | grep "x-ms-thread-id" | cut -d' ' -f2) +echo "Thread ID: $threadId" +``` + +PowerShell: + +```powershell +Invoke-RestMethod -Method Post ` + -Uri http://localhost:7071/api/agents/publisher/run ` + -ResponseHeadersVariable ResponseHeaders ` + -ContentType text/plain ` + -Body 'Start a content generation workflow for the topic \"The Future of Artificial Intelligence\"' ` + +# Save the thread ID to a variable and print it to the console +$threadId = $ResponseHeaders['x-ms-thread-id'] +Write-Host "Thread ID: $threadId" +``` + +The response will be a text string that looks something like the following, indicating that the agent request has been received and will be processed: + +```http +HTTP/1.1 200 OK +Content-Type: text/plain +x-ms-thread-id: 351ec855-7f4d-4527-a60d-498301ced36d + +The content generation workflow for the topic "The Future of Artificial Intelligence" has been successfully started, and the instance ID is **6a04276e8d824d8d941e1dc4142cc254**. If you need any further assistance or updates on the workflow, feel free to ask! +``` + +The `x-ms-thread-id` response header contains the thread ID, which can be used to continue the conversation by passing it as a query parameter (`thread_id`) to the `run` endpoint. The commands above show how to save the thread ID to a `$threadId` variable for use in subsequent requests. + +Behind the scenes, the publisher agent will: + +1. Start the content generation workflow via a tool call +1. The workflow will generate initial content using the Writer agent and wait for human approval, which will be visible in the logs + +Once the workflow is waiting for human approval, you can send approval or rejection by prompting the publisher agent accordingly (e.g. "Approve the content" or "Reject the content with feedback: The article needs more technical depth and better examples."): + +Bash (Linux/macOS/WSL): + +```bash +# Approve the content +curl -X POST "http://localhost:7071/api/agents/publisher/run?thread_id=$threadId" \ + -H "Content-Type: text/plain" \ + -d 'Approve the content' + +# Reject the content with feedback +curl -X POST "http://localhost:7071/api/agents/publisher/run?thread_id=$threadId" \ + -H "Content-Type: text/plain" \ + -d 'Reject the content with feedback: The article needs more technical depth and better examples.' +``` + +PowerShell: + +```powershell +# Approve the content +Invoke-RestMethod -Method Post ` + -Uri "http://localhost:7071/api/agents/publisher/run?thread_id=$threadId" ` + -ContentType text/plain ` + -Body 'Approve the content' + +# Reject the content with feedback +Invoke-RestMethod -Method Post ` + -Uri "http://localhost:7071/api/agents/publisher/run?thread_id=$threadId" ` + -ContentType text/plain ` + -Body 'Reject the content with feedback: The article needs more technical depth and better examples.' +``` + +Once the workflow has completed, you can get the status by prompting the publisher agent to give you the status. + +Bash (Linux/macOS/WSL): + +```bash +curl -X POST "http://localhost:7071/api/agents/publisher/run?thread_id=$threadId" \ + -H "Content-Type: text/plain" \ + -d 'Get the status of the workflow you previously started' +``` + +PowerShell: + +```powershell +Invoke-RestMethod -Method Post ` + -Uri "http://localhost:7071/api/agents/publisher/run?thread_id=$threadId" ` + -ContentType text/plain ` + -Body 'Get the status of the workflow you previously started' +``` + +The response from the publisher agent will look something like the following: + +```text +The status of the workflow with instance ID **ab1076d6e7ec49d8a2c2474d09b69ded** is as follows: + +- **Execution Status:** Completed +- **Workflow Status:** Content published successfully at `2025-10-24T20:42:02` +- **Created At:** `2025-10-24T20:41:40.7531781+00:00` +- **Last Updated At:** `2025-10-24T20:42:02.1410736+00:00` + +The content has been successfully published. +``` diff --git a/dotnet/samples/AzureFunctions/06_LongRunningTools/Tools.cs b/dotnet/samples/AzureFunctions/06_LongRunningTools/Tools.cs new file mode 100644 index 0000000000..c2602e659e --- /dev/null +++ b/dotnet/samples/AzureFunctions/06_LongRunningTools/Tools.cs @@ -0,0 +1,84 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.ComponentModel; +using Microsoft.Agents.AI.DurableTask; +using Microsoft.DurableTask.Client; +using Microsoft.Extensions.Logging; + +namespace LongRunningTools; + +/// +/// Tools that demonstrate starting orchestrations from agent tool calls. +/// +internal sealed class Tools(ILogger logger) +{ + private readonly ILogger _logger = logger; + + [Description("Starts a content generation workflow and returns the instance ID for tracking.")] + public string StartContentGenerationWorkflow([Description("The topic for content generation")] string topic) + { + this._logger.LogInformation("Starting content generation workflow for topic: {Topic}", topic); + + const int MaxReviewAttempts = 3; + const float ApprovalTimeoutHours = 72; + + // Schedule the orchestration, which will start running after the tool call completes. + string instanceId = DurableAgentContext.Current.ScheduleNewOrchestration( + name: nameof(FunctionTriggers.RunOrchestrationAsync), + input: new ContentGenerationInput + { + Topic = topic, + MaxReviewAttempts = MaxReviewAttempts, + ApprovalTimeoutHours = ApprovalTimeoutHours + }); + + this._logger.LogInformation( + "Content generation workflow scheduled to be started for topic '{Topic}' with instance ID: {InstanceId}", + topic, + instanceId); + + return $"Workflow started with instance ID: {instanceId}"; + } + + [Description("Gets the status of a workflow orchestration.")] + public async Task GetWorkflowStatusAsync( + [Description("The instance ID of the workflow to check")] string instanceId, + [Description("Whether to include detailed information")] bool includeDetails = true) + { + this._logger.LogInformation("Getting status for workflow instance: {InstanceId}", instanceId); + + // Get the current agent context using the thread-static property + OrchestrationMetadata? status = await DurableAgentContext.Current.GetOrchestrationStatusAsync( + instanceId, + includeDetails); + + if (status is null) + { + this._logger.LogInformation("Workflow instance '{InstanceId}' not found.", instanceId); + return new + { + instanceId, + error = $"Workflow instance '{instanceId}' not found.", + }; + } + + return new + { + instanceId = status.InstanceId, + createdAt = status.CreatedAt, + executionStatus = status.RuntimeStatus, + workflowStatus = status.SerializedCustomStatus, + lastUpdatedAt = status.LastUpdatedAt, + failureDetails = status.FailureDetails + }; + } + + [Description("Raises a feedback event for the content generation workflow.")] + public async Task SubmitHumanApprovalAsync( + [Description("The instance ID of the workflow to submit feedback for")] string instanceId, + [Description("Feedback to submit")] HumanApprovalResponse feedback) + { + this._logger.LogInformation("Submitting human approval for workflow instance: {InstanceId}", instanceId); + await DurableAgentContext.Current.RaiseOrchestrationEventAsync(instanceId, "HumanApproval", feedback); + } +} diff --git a/dotnet/samples/AzureFunctions/06_LongRunningTools/demo.http b/dotnet/samples/AzureFunctions/06_LongRunningTools/demo.http new file mode 100644 index 0000000000..c0f13f1992 --- /dev/null +++ b/dotnet/samples/AzureFunctions/06_LongRunningTools/demo.http @@ -0,0 +1,27 @@ +### Run an agent that can schedule orchestrations as tool calls +POST http://localhost:7071/api/agents/publisher/run +Content-Type: text/plain + +Start a content generation workflow for the topic 'The Future of Artificial Intelligence' + + +### Save the session ID from the response to continue the conversation +@threadId = + +### Check the status of the workflow +POST http://localhost:7071/api/agents/publisher/run?thread_id={{threadId}} +Content-Type: text/plain + +Check the status of the workflow you previously started + +### Reject content with feedback +POST http://localhost:7071/api/agents/publisher/run?thread_id={{threadId}} +Content-Type: text/plain + +Reject the content with feedback: The article needs more technical depth and better examples. + +### Approve content +POST http://localhost:7071/api/agents/publisher/run?thread_id={{threadId}} +Content-Type: text/plain + +Approve the content diff --git a/dotnet/samples/AzureFunctions/06_LongRunningTools/host.json b/dotnet/samples/AzureFunctions/06_LongRunningTools/host.json new file mode 100644 index 0000000000..9384a0a583 --- /dev/null +++ b/dotnet/samples/AzureFunctions/06_LongRunningTools/host.json @@ -0,0 +1,20 @@ +{ + "version": "2.0", + "logging": { + "logLevel": { + "Microsoft.Agents.AI.DurableTask": "Information", + "Microsoft.Agents.AI.Hosting.AzureFunctions": "Information", + "DurableTask": "Information", + "Microsoft.DurableTask": "Information" + } + }, + "extensions": { + "durableTask": { + "hubName": "default", + "storageProvider": { + "type": "AzureManaged", + "connectionStringName": "DURABLE_TASK_SCHEDULER_CONNECTION_STRING" + } + } + } +} diff --git a/dotnet/samples/AzureFunctions/06_LongRunningTools/local.settings.json b/dotnet/samples/AzureFunctions/06_LongRunningTools/local.settings.json new file mode 100644 index 0000000000..54dfbb5664 --- /dev/null +++ b/dotnet/samples/AzureFunctions/06_LongRunningTools/local.settings.json @@ -0,0 +1,10 @@ +{ + "IsEncrypted": false, + "Values": { + "FUNCTIONS_WORKER_RUNTIME": "dotnet-isolated", + "AzureWebJobsStorage": "UseDevelopmentStorage=true", + "DURABLE_TASK_SCHEDULER_CONNECTION_STRING": "Endpoint=http://localhost:8080;TaskHub=default;Authentication=None", + "AZURE_OPENAI_ENDPOINT": "", + "AZURE_OPENAI_DEPLOYMENT": "" + } +} diff --git a/dotnet/samples/AzureFunctions/07_AgentAsMcpTool/07_AgentAsMcpTool.csproj b/dotnet/samples/AzureFunctions/07_AgentAsMcpTool/07_AgentAsMcpTool.csproj new file mode 100644 index 0000000000..12795b2efb --- /dev/null +++ b/dotnet/samples/AzureFunctions/07_AgentAsMcpTool/07_AgentAsMcpTool.csproj @@ -0,0 +1,42 @@ + + + net10.0 + v4 + Exe + enable + enable + + AgentAsMcpTool + AgentAsMcpTool + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/dotnet/samples/AzureFunctions/07_AgentAsMcpTool/Program.cs b/dotnet/samples/AzureFunctions/07_AgentAsMcpTool/Program.cs new file mode 100644 index 0000000000..bc0a69cbf2 --- /dev/null +++ b/dotnet/samples/AzureFunctions/07_AgentAsMcpTool/Program.cs @@ -0,0 +1,53 @@ +// Copyright (c) Microsoft. All rights reserved. + +// This sample demonstrates how to configure AI agents to be accessible as MCP tools. +// When using AddAIAgent and enabling MCP tool triggers, the Functions host will automatically +// generate a remote MCP endpoint for the app at /runtime/webhooks/mcp with a agent-specific +// query tool name. + +using Azure; +using Azure.AI.OpenAI; +using Azure.Identity; +using Microsoft.Agents.AI; +using Microsoft.Agents.AI.DurableTask; +using Microsoft.Agents.AI.Hosting.AzureFunctions; +using Microsoft.Azure.Functions.Worker.Builder; +using Microsoft.Extensions.Hosting; +using OpenAI.Chat; + +// Get the Azure OpenAI endpoint and deployment name from environment variables. +string endpoint = Environment.GetEnvironmentVariable("AZURE_OPENAI_ENDPOINT") + ?? throw new InvalidOperationException("AZURE_OPENAI_ENDPOINT is not set."); +string deploymentName = Environment.GetEnvironmentVariable("AZURE_OPENAI_DEPLOYMENT") + ?? throw new InvalidOperationException("AZURE_OPENAI_DEPLOYMENT is not set."); + +// Use Azure Key Credential if provided, otherwise use Azure CLI Credential. +string? azureOpenAiKey = System.Environment.GetEnvironmentVariable("AZURE_OPENAI_KEY"); +AzureOpenAIClient client = !string.IsNullOrEmpty(azureOpenAiKey) + ? new AzureOpenAIClient(new Uri(endpoint), new AzureKeyCredential(azureOpenAiKey)) + : new AzureOpenAIClient(new Uri(endpoint), new AzureCliCredential()); + +// Define three AI agents we are going to use in this application. +AIAgent agent1 = client.GetChatClient(deploymentName).CreateAIAgent("You are good at telling jokes.", "Joker"); + +AIAgent agent2 = client.GetChatClient(deploymentName) + .CreateAIAgent("Check stock prices.", "StockAdvisor"); + +AIAgent agent3 = client.GetChatClient(deploymentName) + .CreateAIAgent("Recommend plants.", "PlantAdvisor", description: "Get plant recommendations."); + +using IHost app = FunctionsApplication + .CreateBuilder(args) + .ConfigureFunctionsWebApplication() + .ConfigureDurableAgents(options => + { + options + .AddAIAgent(agent1) // Enables HTTP trigger by default. + .AddAIAgent(agent2, enableHttpTrigger: false, enableMcpToolTrigger: true) // Disable HTTP trigger, enable MCP Tool trigger. + .AddAIAgent(agent3, agentOptions => + { + agentOptions.McpToolTrigger.IsEnabled = true; // Enable MCP Tool trigger. + }); + }) + .Build(); +app.Run(); diff --git a/dotnet/samples/AzureFunctions/07_AgentAsMcpTool/README.md b/dotnet/samples/AzureFunctions/07_AgentAsMcpTool/README.md new file mode 100644 index 0000000000..a8efad04de --- /dev/null +++ b/dotnet/samples/AzureFunctions/07_AgentAsMcpTool/README.md @@ -0,0 +1,87 @@ +# Agent as MCP Tool Sample + +This sample demonstrates how to configure AI agents to be accessible as both HTTP endpoints and [Model Context Protocol (MCP)](https://modelcontextprotocol.io/) tools, enabling flexible integration patterns for AI agent consumption. + +## Key Concepts Demonstrated + +- **Multi-trigger Agent Configuration**: Configure agents to support HTTP triggers, MCP tool triggers, or both +- **Microsoft Agent Framework Integration**: Use the framework to define AI agents with specific roles and capabilities +- **Flexible Agent Registration**: Register agents with customizable trigger configurations +- **MCP Server Hosting**: Expose agents as MCP tools for consumption by MCP-compatible clients + +## Sample Architecture + +This sample creates three agents with different trigger configurations: + +| Agent | Role | HTTP Trigger | MCP Tool Trigger | Description | +|-------|------|--------------|------------------|-------------| +| **Joker** | Comedy specialist | ✅ Enabled | ❌ Disabled | Accessible only via HTTP requests | +| **StockAdvisor** | Financial data | ❌ Disabled | ✅ Enabled | Accessible only as MCP tool | +| **PlantAdvisor** | Indoor plant recommendations | ✅ Enabled | ✅ Enabled | Accessible via both HTTP and MCP | + +## Environment Setup + +See the [README.md](../README.md) file in the parent directory for complete setup instructions, including: + +- Prerequisites installation +- Azure OpenAI configuration +- Durable Task Scheduler setup +- Storage emulator configuration + +For this sample, you'll also need to install [node.js](https://nodejs.org/en/download) in order to use the [MCP Inspector](https://modelcontextprotocol.io/docs/tools/inspector) tool. + +## Configuration + +Update your `local.settings.json` with your Azure OpenAI credentials: + +```json +{ + "Values": { + "AZURE_OPENAI_ENDPOINT": "https://your-resource.openai.azure.com/", + "AZURE_OPENAI_DEPLOYMENT": "your-deployment-name", + "AZURE_OPENAI_KEY": "your-api-key-if-not-using-rbac" + } +} +``` + +## Running the Sample + +1. **Start the Function App**: + + ```bash + cd dotnet/samples/AzureFunctions/07_AgentAsMcpTool + func start + ``` + +2. **Note the MCP Server Endpoint**: When the app starts, you'll see the MCP server endpoint in the terminal output. It will look like: + + ```text + MCP server endpoint: http://localhost:7071/runtime/webhooks/mcp + ``` + +## Testing MCP Tool Integration + +Any MCP-compatible client can connect to the server endpoint and utilize the exposed agent tools. The agents will appear as callable tools within the MCP protocol. + +### Using MCP Inspector + +1. Run the [MCP Inspector](https://modelcontextprotocol.io/docs/tools/inspector) from the command line: + + ```bash + npx @modelcontextprotocol/inspector + ``` + +1. Connect using the MCP server endpoint from your terminal output + + - For **Transport Type**, select **"Streamable HTTP"** + - For **URL**, enter the MCP server endpoint `http://localhost:7071/runtime/webhooks/mcp` + - Click the **Connect** button + +1. Click the **List Tools** button to see the available MCP tools. You should see the `StockAdvisor` and `PlantAdvisor` tools. + +1. Test the available MCP tools: + + - **StockAdvisor** - Set "MSFT ATH" (ATH is "all time high") as the query and click the **Run Tool** button. + - **PlantAdvisor** - Set "Low light in Seattle" as the query and click the **Run Tool** button. + +You'll see the results of the tool calls in the MCP Inspector interface under the **Tool Results** section. You should also see the results in the terminal where you ran the `func start` command. diff --git a/dotnet/samples/AzureFunctions/07_AgentAsMcpTool/host.json b/dotnet/samples/AzureFunctions/07_AgentAsMcpTool/host.json new file mode 100644 index 0000000000..aa36d82912 --- /dev/null +++ b/dotnet/samples/AzureFunctions/07_AgentAsMcpTool/host.json @@ -0,0 +1,19 @@ +{ + "version": "2.0", + "logging": { + "logLevel": { + "Microsoft.Azure.Functions.DurableAgents": "Information", + "DurableTask": "Information", + "Microsoft.DurableTask": "Information" + } + }, + "extensions": { + "durableTask": { + "hubName": "default", + "storageProvider": { + "type": "AzureManaged", + "connectionStringName": "DURABLE_TASK_SCHEDULER_CONNECTION_STRING" + } + } + } +} diff --git a/dotnet/samples/AzureFunctions/07_AgentAsMcpTool/local.settings.json b/dotnet/samples/AzureFunctions/07_AgentAsMcpTool/local.settings.json new file mode 100644 index 0000000000..54dfbb5664 --- /dev/null +++ b/dotnet/samples/AzureFunctions/07_AgentAsMcpTool/local.settings.json @@ -0,0 +1,10 @@ +{ + "IsEncrypted": false, + "Values": { + "FUNCTIONS_WORKER_RUNTIME": "dotnet-isolated", + "AzureWebJobsStorage": "UseDevelopmentStorage=true", + "DURABLE_TASK_SCHEDULER_CONNECTION_STRING": "Endpoint=http://localhost:8080;TaskHub=default;Authentication=None", + "AZURE_OPENAI_ENDPOINT": "", + "AZURE_OPENAI_DEPLOYMENT": "" + } +} diff --git a/dotnet/samples/AzureFunctions/README.md b/dotnet/samples/AzureFunctions/README.md new file mode 100644 index 0000000000..e60b0f662e --- /dev/null +++ b/dotnet/samples/AzureFunctions/README.md @@ -0,0 +1,151 @@ +# Azure Functions Samples + +This directory contains samples for Azure Functions. + +- **[01_SingleAgent](01_SingleAgent)**: A sample that demonstrates how to host a single conversational agent in an Azure Functions app and invoke it directly over HTTP. +- **[02_AgentOrchestration_Chaining](02_AgentOrchestration_Chaining)**: A sample that demonstrates how to host a single conversational agent in an Azure Functions app and invoke it using a durable orchestration. +- **[03_AgentOrchestration_Concurrency](03_AgentOrchestration_Concurrency)**: A sample that demonstrates how to host multiple agents in an Azure Functions app and run them concurrently using a durable orchestration. +- **[04_AgentOrchestration_Conditionals](04_AgentOrchestration_Conditionals)**: A sample that demonstrates how to host multiple agents in an Azure Functions app and run them sequentially using a durable orchestration with conditionals. +- **[05_AgentOrchestration_HITL](05_AgentOrchestration_HITL)**: A sample that demonstrates how to implement a human-in-the-loop workflow using durable orchestration, including external event handling for human approval. +- **[06_LongRunningTools](06_LongRunningTools)**: A sample that demonstrates how agents can start and interact with durable orchestrations from tool calls to enable long-running tool scenarios. +- **[07_AgentAsMcpTool](07_AgentAsMcpTool)**: A sample that demonstrates how to configure durable AI agents to be accessible as Model Context Protocol (MCP) tools. + +## Running the Samples + +These samples are designed to be run locally in a cloned repository. + +### Prerequisites + +The following prerequisites are required to run the samples: + +- [.NET 10.0 SDK or later](https://dotnet.microsoft.com/download/dotnet) +- [Azure Functions Core Tools](https://learn.microsoft.com/azure/azure-functions/functions-run-local) (version 4.x or later) +- [Azure CLI](https://learn.microsoft.com/cli/azure/install-azure-cli) installed and authenticated (`az login`) or an API key for the Azure OpenAI service +- [Azure OpenAI Service](https://learn.microsoft.com/azure/ai-services/openai/how-to/create-resource) with a deployed model (gpt-4o-mini or better is recommended) +- [Durable Task Scheduler](https://learn.microsoft.com/azure/azure-functions/durable/durable-task-scheduler/develop-with-durable-task-scheduler) (local emulator or Azure-hosted) +- [Docker](https://docs.docker.com/get-docker/) installed if running the Durable Task Scheduler emulator locally + +### Configuring RBAC Permissions for Azure OpenAI + +These samples are configured to use the Azure OpenAI service with RBAC permissions to access the model. You'll need to configure the RBAC permissions for the Azure OpenAI service to allow the Azure Functions app to access the model. + +Below is an example of how to configure the RBAC permissions for the Azure OpenAI service to allow the current user to access the model. + +Bash (Linux/macOS/WSL): + +```bash +az role assignment create \ + --assignee "yourname@contoso.com" \ + --role "Cognitive Services OpenAI User" \ + --scope /subscriptions//resourceGroups//providers/Microsoft.CognitiveServices/accounts/ +``` + +PowerShell: + +```powershell +az role assignment create ` + --assignee "yourname@contoso.com" ` + --role "Cognitive Services OpenAI User" ` + --scope /subscriptions//resourceGroups//providers/Microsoft.CognitiveServices/accounts/ +``` + +More information on how to configure RBAC permissions for Azure OpenAI can be found in the [Azure OpenAI documentation](https://learn.microsoft.com/azure/ai-services/openai/how-to/create-resource?pivots=cli). + +### Setting an API key for the Azure OpenAI service + +As an alternative to configuring Azure RBAC permissions, you can set an API key for the Azure OpenAI service by setting the `AZURE_OPENAI_KEY` environment variable. + +Bash (Linux/macOS/WSL): + +```bash +export AZURE_OPENAI_KEY="your-api-key" +``` + +PowerShell: + +```powershell +$env:AZURE_OPENAI_KEY="your-api-key" +``` + +### Start Durable Task Scheduler + +Most samples use the Durable Task Scheduler (DTS) to support hosted agents and durable orchestrations. DTS also allows you to view the status of orchestrations and their inputs and outputs from a web UI. + +To run the Durable Task Scheduler locally, you can use the following `docker` command: + +```bash +docker run -d --name dts-emulator -p 8080:8080 -p 8082:8082 mcr.microsoft.com/dts/dts-emulator:latest +``` + +The DTS dashboard will be available at `http://localhost:8080`. + +### Start the Azure Storage Emulator + +All Function apps require an Azure Storage account to store functions-specific state. You can use the Azure Storage Emulator to run a local instance of the Azure Storage service. + +You can run the Azure Storage emulator locally as a standalone process or via a Docker container. + +#### Docker + +```bash +docker run -d --name storage-emulator -p 10000:10000 -p 10001:10001 -p 10002:10002 mcr.microsoft.com/azure-storage/azurite +``` + +#### Standalone + +```bash +npm install -g azurite +azurite +``` + +### Environment Configuration + +Each sample has its own `local.settings.json` file that contains the environment variables for the sample. You'll need to update the `local.settings.json` file with the correct values for your Azure OpenAI resource. + +```json +{ + "Values": { + "AZURE_OPENAI_ENDPOINT": "https://your-resource.openai.azure.com/", + "AZURE_OPENAI_DEPLOYMENT": "your-deployment-name" + } +} +``` + +Alternatively, you can set the environment variables in the command line. + +### Bash (Linux/macOS/WSL) + +```bash +export AZURE_OPENAI_ENDPOINT="https://your-resource.openai.azure.com/" +export AZURE_OPENAI_DEPLOYMENT="your-deployment-name" +``` + +### PowerShell + +```powershell +$env:AZURE_OPENAI_ENDPOINT="https://your-resource.openai.azure.com/" +$env:AZURE_OPENAI_DEPLOYMENT="your-deployment-name" +``` + +These environment variables, when set, will override the values in the `local.settings.json` file, making it convenient to test the sample without having to update the `local.settings.json` file. + +### Start the Azure Functions app + +Navigate to the sample directory and start the Azure Functions app: + +```bash +cd dotnet/samples/AzureFunctions/01_SingleAgent +func start +``` + +The Azure Functions app will be available at `http://localhost:7071`. + +### Test the Azure Functions app + +The README.md file in each sample directory contains instructions for testing the sample. Each sample also includes a `demo.http` file that can be used to test the sample from the command line. These files can be opened in VS Code with the [REST Client](https://marketplace.visualstudio.com/items?itemName=humao.rest-client) extension or in the Visual Studio IDE. + +### Viewing the sample output + +The Azure Functions app logs are displayed in the terminal where you ran `func start`. This is where most agent output will be displayed. You can adjust logging levels in the `host.json` file as needed. + +You can also see the state of agents and orchestrations in the DTS dashboard. diff --git a/dotnet/samples/Directory.Build.props b/dotnet/samples/Directory.Build.props index dd86677c3e..15880d4a8e 100644 --- a/dotnet/samples/Directory.Build.props +++ b/dotnet/samples/Directory.Build.props @@ -5,7 +5,7 @@ false false - net472;net9.0 + net10.0;net472 5ee045b0-aea3-4f08-8d31-32d1a6f8fed0 diff --git a/dotnet/samples/GettingStarted/A2A/A2AAgent_AsFunctionTools/A2AAgent_AsFunctionTools.csproj b/dotnet/samples/GettingStarted/A2A/A2AAgent_AsFunctionTools/A2AAgent_AsFunctionTools.csproj index 2b89b20fbf..d91b20e34b 100644 --- a/dotnet/samples/GettingStarted/A2A/A2AAgent_AsFunctionTools/A2AAgent_AsFunctionTools.csproj +++ b/dotnet/samples/GettingStarted/A2A/A2AAgent_AsFunctionTools/A2AAgent_AsFunctionTools.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable @@ -13,8 +13,6 @@ - - diff --git a/dotnet/samples/GettingStarted/A2A/A2AAgent_AsFunctionTools/Program.cs b/dotnet/samples/GettingStarted/A2A/A2AAgent_AsFunctionTools/Program.cs index a6f701cde3..c813464d0c 100644 --- a/dotnet/samples/GettingStarted/A2A/A2AAgent_AsFunctionTools/Program.cs +++ b/dotnet/samples/GettingStarted/A2A/A2AAgent_AsFunctionTools/Program.cs @@ -10,7 +10,7 @@ using Azure.Identity; using Microsoft.Agents.AI; using Microsoft.Extensions.AI; -using OpenAI; +using OpenAI.Chat; var endpoint = Environment.GetEnvironmentVariable("AZURE_OPENAI_ENDPOINT") ?? throw new InvalidOperationException("AZURE_OPENAI_ENDPOINT is not set."); var deploymentName = Environment.GetEnvironmentVariable("AZURE_OPENAI_DEPLOYMENT_NAME") ?? "gpt-4o-mini"; diff --git a/dotnet/samples/GettingStarted/A2A/A2AAgent_AsFunctionTools/README.md b/dotnet/samples/GettingStarted/A2A/A2AAgent_AsFunctionTools/README.md index 6cbd56dca4..c050ad0830 100644 --- a/dotnet/samples/GettingStarted/A2A/A2AAgent_AsFunctionTools/README.md +++ b/dotnet/samples/GettingStarted/A2A/A2AAgent_AsFunctionTools/README.md @@ -7,7 +7,7 @@ and register these function tools with another AI agent so it can leverage the A Before you begin, ensure you have the following prerequisites: -- .NET 8.0 SDK or later +- .NET 10 SDK or later - Access to the A2A agent host service **Note**: These samples need to be run against a valid A2A server. If no A2A server is available, they can be run against the echo-agent that can be diff --git a/dotnet/samples/GettingStarted/Agents/Agent_Step03.2_UsingFunctionTools_FromOpenAPI/Agent_Step03.2_UsingFunctionTools_FromOpenAPI.csproj b/dotnet/samples/GettingStarted/A2A/A2AAgent_PollingForTaskCompletion/A2AAgent_PollingForTaskCompletion.csproj similarity index 50% rename from dotnet/samples/GettingStarted/Agents/Agent_Step03.2_UsingFunctionTools_FromOpenAPI/Agent_Step03.2_UsingFunctionTools_FromOpenAPI.csproj rename to dotnet/samples/GettingStarted/A2A/A2AAgent_PollingForTaskCompletion/A2AAgent_PollingForTaskCompletion.csproj index e2edbb2f8d..1f36cef576 100644 --- a/dotnet/samples/GettingStarted/Agents/Agent_Step03.2_UsingFunctionTools_FromOpenAPI/Agent_Step03.2_UsingFunctionTools_FromOpenAPI.csproj +++ b/dotnet/samples/GettingStarted/A2A/A2AAgent_PollingForTaskCompletion/A2AAgent_PollingForTaskCompletion.csproj @@ -1,28 +1,25 @@ - + Exe - net9.0 + net10.0 enable enable + - - + + + + - - - PreserveNewest - - - diff --git a/dotnet/samples/GettingStarted/A2A/A2AAgent_PollingForTaskCompletion/Program.cs b/dotnet/samples/GettingStarted/A2A/A2AAgent_PollingForTaskCompletion/Program.cs new file mode 100644 index 0000000000..7b5934575c --- /dev/null +++ b/dotnet/samples/GettingStarted/A2A/A2AAgent_PollingForTaskCompletion/Program.cs @@ -0,0 +1,35 @@ +// Copyright (c) Microsoft. All rights reserved. + +// This sample demonstrates how to poll for long-running task completion using continuation tokens with an A2A AI agent. + +using A2A; +using Microsoft.Agents.AI; + +var a2aAgentHost = Environment.GetEnvironmentVariable("A2A_AGENT_HOST") ?? throw new InvalidOperationException("A2A_AGENT_HOST is not set."); + +// Initialize an A2ACardResolver to get an A2A agent card. +A2ACardResolver agentCardResolver = new(new Uri(a2aAgentHost)); + +// Get the agent card +AgentCard agentCard = await agentCardResolver.GetAgentCardAsync(); + +// Create an instance of the AIAgent for an existing A2A agent specified by the agent card. +AIAgent agent = agentCard.GetAIAgent(); + +AgentThread thread = agent.GetNewThread(); + +// Start the initial run with a long-running task. +AgentRunResponse response = await agent.RunAsync("Conduct a comprehensive analysis of quantum computing applications in cryptography, including recent breakthroughs, implementation challenges, and future roadmap. Please include diagrams and visual representations to illustrate complex concepts.", thread); + +// Poll until the response is complete. +while (response.ContinuationToken is { } token) +{ + // Wait before polling again. + await Task.Delay(TimeSpan.FromSeconds(2)); + + // Continue with the token. + response = await agent.RunAsync(thread, options: new AgentRunOptions { ContinuationToken = token }); +} + +// Display the result +Console.WriteLine(response); diff --git a/dotnet/samples/GettingStarted/A2A/A2AAgent_PollingForTaskCompletion/README.md b/dotnet/samples/GettingStarted/A2A/A2AAgent_PollingForTaskCompletion/README.md new file mode 100644 index 0000000000..3e1160b510 --- /dev/null +++ b/dotnet/samples/GettingStarted/A2A/A2AAgent_PollingForTaskCompletion/README.md @@ -0,0 +1,25 @@ +# Polling for A2A Agent Task Completion + +This sample demonstrates how to poll for long-running task completion using continuation tokens with an A2A AI agent, following the background responses pattern. + +The sample: + +- Connects to an A2A agent server specified in the `A2A_AGENT_HOST` environment variable +- Sends a request to the agent that may take time to complete +- Polls the agent at regular intervals using continuation tokens until a final response is received +- Displays the final result + +This pattern is useful when an AI model cannot complete a complex task in a single response and needs multiple rounds of processing. + +# Prerequisites + +Before you begin, ensure you have the following prerequisites: + +- .NET 10.0 SDK or later +- An A2A agent server running and accessible via HTTP + +Set the following environment variable: + +```powershell +$env:A2A_AGENT_HOST="http://localhost:5000" # Replace with your A2A agent server host +``` diff --git a/dotnet/samples/GettingStarted/A2A/README.md b/dotnet/samples/GettingStarted/A2A/README.md index 3ddac95996..b513ffa929 100644 --- a/dotnet/samples/GettingStarted/A2A/README.md +++ b/dotnet/samples/GettingStarted/A2A/README.md @@ -14,6 +14,7 @@ See the README.md for each sample for the prerequisites for that sample. |Sample|Description| |---|---| |[A2A Agent As Function Tools](./A2AAgent_AsFunctionTools/)|This sample demonstrates how to represent an A2A agent as a set of function tools, where each function tool corresponds to a skill of the A2A agent, and register these function tools with another AI agent so it can leverage the A2A agent's skills.| +|[A2A Agent Polling For Task Completion](./A2AAgent_PollingForTaskCompletion/)|This sample demonstrates how to poll for long-running task completion using continuation tokens with an A2A agent.| ## Running the samples from the console diff --git a/dotnet/samples/GettingStarted/AGUI/README.md b/dotnet/samples/GettingStarted/AGUI/README.md new file mode 100644 index 0000000000..a624fe81f3 --- /dev/null +++ b/dotnet/samples/GettingStarted/AGUI/README.md @@ -0,0 +1,304 @@ +# AG-UI Getting Started Samples + +This directory contains samples that demonstrate how to build AG-UI (Agent UI Protocol) servers and clients using the Microsoft Agent Framework. + +## Prerequisites + +- .NET 9.0 or later +- Azure OpenAI service endpoint and deployment configured +- Azure CLI installed and authenticated (`az login`) +- User has the `Cognitive Services OpenAI Contributor` role for the Azure OpenAI resource + +## Environment Variables + +All samples require the following environment variables: + +```bash +export AZURE_OPENAI_ENDPOINT="https://your-resource.openai.azure.com/" +export AZURE_OPENAI_DEPLOYMENT_NAME="gpt-4o-mini" +``` + +For the client samples, you can optionally set: + +```bash +export AGUI_SERVER_URL="http://localhost:8888" +``` + +## Samples + +### Step01_GettingStarted + +A basic AG-UI server and client that demonstrate the foundational concepts. + +#### Server (`Step01_GettingStarted/Server`) + +A basic AG-UI server that hosts an AI agent accessible via HTTP. Demonstrates: + +- Creating an ASP.NET Core web application +- Setting up an AG-UI server endpoint with `MapAGUI` +- Creating an AI agent from an Azure OpenAI chat client +- Streaming responses via Server-Sent Events (SSE) + +**Run the server:** + +```bash +cd Step01_GettingStarted/Server +dotnet run --urls http://localhost:8888 +``` + +#### Client (`Step01_GettingStarted/Client`) + +An interactive console client that connects to an AG-UI server. Demonstrates: + +- Creating an AG-UI client with `AGUIChatClient` +- Managing conversation threads +- Streaming responses with `RunStreamingAsync` +- Displaying colored console output for different content types +- Supporting both interactive and automated modes + +**Prerequisites:** The Step01_GettingStarted server (or any AG-UI server) must be running. + +**Run the client:** + +```bash +cd Step01_GettingStarted/Client +dotnet run +``` + +Type messages and press Enter to interact with the agent. Type `:q` or `quit` to exit. + +### Step02_BackendTools + +An AG-UI server with function tools that execute on the backend. + +#### Server (`Step02_BackendTools/Server`) + +Demonstrates: + +- Creating function tools using `AIFunctionFactory.Create` +- Using `[Description]` attributes for tool documentation +- Defining explicit request/response types for type safety +- Setting up JSON serialization contexts for source generation +- Backend tool rendering (tools execute on the server) + +**Run the server:** + +```bash +cd Step02_BackendTools/Server +dotnet run --urls http://localhost:8888 +``` + +#### Client (`Step02_BackendTools/Client`) + +A client that works with the backend tools server. Try asking: "Find Italian restaurants in Seattle" or "Search for Mexican food in Portland". + +**Run the client:** + +```bash +cd Step02_BackendTools/Client +dotnet run +``` + +### Step03_FrontendTools + +Demonstrates frontend tool rendering (tools defined on client, executed on server). + +#### Server (`Step03_FrontendTools/Server`) + +A basic AG-UI server that accepts tool definitions from the client. + +**Run the server:** + +```bash +cd Step03_FrontendTools/Server +dotnet run --urls http://localhost:8888 +``` + +#### Client (`Step03_FrontendTools/Client`) + +A client that defines and sends tools to the server for execution. + +**Run the client:** + +```bash +cd Step03_FrontendTools/Client +dotnet run +``` + +### Step04_HumanInLoop + +Demonstrates human-in-the-loop approval workflows for sensitive operations. This sample includes both a server and client component. + +#### Server (`Step04_HumanInLoop/Server`) + +An AG-UI server that implements approval workflows. Demonstrates: + +- Wrapping tools with `ApprovalRequiredAIFunction` +- Converting `FunctionApprovalRequestContent` to approval requests +- Middleware pattern with `ServerFunctionApprovalServerAgent` +- Complete function call capture and restoration + +**Run the server:** + +```bash +cd Step04_HumanInLoop/Server +dotnet run --urls http://localhost:8888 +``` + +#### Client (`Step04_HumanInLoop/Client`) + +An interactive client that handles approval requests from the server. Demonstrates: + +- Using `ServerFunctionApprovalClientAgent` middleware +- Detecting `FunctionApprovalRequestContent` +- Displaying approval details to users +- Prompting for approval/rejection +- Sending approval responses with `FunctionApprovalResponseContent` +- Resuming conversation after approval + +**Run the client:** + +```bash +cd Step04_HumanInLoop/Client +dotnet run +``` + +Try asking the agent to perform sensitive operations like "Approve expense report EXP-12345". + +### Step05_StateManagement + +An AG-UI server and client that demonstrate state management with predictive updates. + +#### Server (`Step05_StateManagement/Server`) + +Demonstrates: + +- Defining state schemas using C# records +- Using `SharedStateAgent` middleware for state management +- Streaming predictive state updates with `AgentState` content +- Managing shared state between client and server +- Using JSON serialization contexts for state types + +**Run the server:** + +```bash +cd Step05_StateManagement/Server +dotnet run +``` + +The server runs on port 8888 by default. + +#### Client (`Step05_StateManagement/Client`) + +A client that displays and updates shared state from the server. Try asking: "Create a recipe for chocolate chip cookies" or "Suggest a pasta dish". + +**Run the client:** + +```bash +cd Step05_StateManagement/Client +dotnet run +``` + +## How AG-UI Works + +### Server-Side + +1. Client sends HTTP POST request with messages +2. ASP.NET Core endpoint receives the request via `MapAGUI` +3. Agent processes messages using Agent Framework +4. Responses are streamed back as Server-Sent Events (SSE) + +### Client-Side + +1. `AGUIAgent` sends HTTP POST request to server +2. Server responds with SSE stream +3. Client parses events into `AgentRunResponseUpdate` objects +4. Updates are displayed based on content type +5. `ConversationId` maintains conversation context + +### Protocol Features + +- **HTTP POST** for requests +- **Server-Sent Events (SSE)** for streaming responses +- **JSON** for event serialization +- **Thread IDs** (as `ConversationId`) for conversation context +- **Run IDs** (as `ResponseId`) for tracking individual executions + +## Troubleshooting + +### Connection Refused + +Ensure the server is running before starting the client: + +```bash +# Terminal 1 +cd AGUI_Step01_ServerBasic +dotnet run --urls http://localhost:8888 + +# Terminal 2 (after server starts) +cd AGUI_Step02_ClientBasic +dotnet run +``` + +### Port Already in Use + +If port 8888 is already in use, choose a different port: + +```bash +# Server +dotnet run --urls http://localhost:8889 + +# Client (set environment variable) +export AGUI_SERVER_URL="http://localhost:8889" +dotnet run +``` + +### Authentication Errors + +Make sure you're authenticated with Azure: + +```bash +az login +``` + +Verify you have the `Cognitive Services OpenAI Contributor` role on the Azure OpenAI resource. + +### Missing Environment Variables + +If you see "AZURE_OPENAI_ENDPOINT is not set" errors, ensure environment variables are set in your current shell session before running the samples. + +### Streaming Not Working + +Check that the client timeout is sufficient (default is 60 seconds). For long-running operations, you may need to increase the timeout in the client code. + +## Next Steps + +After completing these samples, explore more AG-UI capabilities: + +### Currently Available in C# + +The samples above demonstrate the AG-UI features currently available in C#: + +- ✅ **Basic Server and Client**: Setting up AG-UI communication +- ✅ **Backend Tool Rendering**: Function tools that execute on the server +- ✅ **Streaming Responses**: Real-time Server-Sent Events +- ✅ **State Management**: State schemas with predictive updates +- ✅ **Human-in-the-Loop**: Approval workflows for sensitive operations + +### Coming Soon to C# + +The following advanced AG-UI features are available in the Python implementation and are planned for future C# releases: + +- ⏳ **Generative UI**: Custom UI component generation +- ⏳ **Advanced State Patterns**: Complex state synchronization scenarios + +For the most up-to-date AG-UI features, see the [Python samples](../../../../python/samples/) for working examples. + +### Related Documentation + +- [AG-UI Overview](https://learn.microsoft.com/agent-framework/integrations/ag-ui/) - Complete AG-UI documentation +- [Getting Started Tutorial](https://learn.microsoft.com/agent-framework/integrations/ag-ui/getting-started) - Step-by-step walkthrough +- [Backend Tool Rendering](https://learn.microsoft.com/agent-framework/integrations/ag-ui/backend-tool-rendering) - Function tools tutorial +- [Human-in-the-Loop](https://learn.microsoft.com/agent-framework/integrations/ag-ui/human-in-the-loop) - Approval workflows tutorial +- [State Management](https://learn.microsoft.com/agent-framework/integrations/ag-ui/state-management) - State management tutorial +- [Agent Framework Overview](https://learn.microsoft.com/agent-framework/overview/agent-framework-overview) - Core framework concepts diff --git a/dotnet/samples/GettingStarted/AGUI/Step01_GettingStarted/Client/Client.csproj b/dotnet/samples/GettingStarted/AGUI/Step01_GettingStarted/Client/Client.csproj new file mode 100644 index 0000000000..a76a2b37ef --- /dev/null +++ b/dotnet/samples/GettingStarted/AGUI/Step01_GettingStarted/Client/Client.csproj @@ -0,0 +1,15 @@ + + + + Exe + net10.0 + enable + enable + + + + + + + + diff --git a/dotnet/samples/GettingStarted/AGUI/Step01_GettingStarted/Client/Program.cs b/dotnet/samples/GettingStarted/AGUI/Step01_GettingStarted/Client/Program.cs new file mode 100644 index 0000000000..d942314806 --- /dev/null +++ b/dotnet/samples/GettingStarted/AGUI/Step01_GettingStarted/Client/Program.cs @@ -0,0 +1,94 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Microsoft.Agents.AI; +using Microsoft.Agents.AI.AGUI; +using Microsoft.Extensions.AI; + +string serverUrl = Environment.GetEnvironmentVariable("AGUI_SERVER_URL") ?? "http://localhost:8888"; + +Console.WriteLine($"Connecting to AG-UI server at: {serverUrl}\n"); + +// Create the AG-UI client agent +using HttpClient httpClient = new() +{ + Timeout = TimeSpan.FromSeconds(60) +}; + +AGUIChatClient chatClient = new(httpClient, serverUrl); + +AIAgent agent = chatClient.CreateAIAgent( + name: "agui-client", + description: "AG-UI Client Agent"); + +AgentThread thread = agent.GetNewThread(); +List messages = +[ + new(ChatRole.System, "You are a helpful assistant.") +]; + +try +{ + while (true) + { + // Get user input + Console.Write("\nUser (:q or quit to exit): "); + string? message = Console.ReadLine(); + + if (string.IsNullOrWhiteSpace(message)) + { + Console.WriteLine("Request cannot be empty."); + continue; + } + + if (message is ":q" or "quit") + { + break; + } + + messages.Add(new ChatMessage(ChatRole.User, message)); + + // Stream the response + bool isFirstUpdate = true; + string? threadId = null; + + await foreach (AgentRunResponseUpdate update in agent.RunStreamingAsync(messages, thread)) + { + ChatResponseUpdate chatUpdate = update.AsChatResponseUpdate(); + + // First update indicates run started + if (isFirstUpdate) + { + threadId = chatUpdate.ConversationId; + Console.ForegroundColor = ConsoleColor.Yellow; + Console.WriteLine($"\n[Run Started - Thread: {chatUpdate.ConversationId}, Run: {chatUpdate.ResponseId}]"); + Console.ResetColor(); + isFirstUpdate = false; + } + + // Display streaming text content + foreach (AIContent content in update.Contents) + { + if (content is TextContent textContent) + { + Console.ForegroundColor = ConsoleColor.Cyan; + Console.Write(textContent.Text); + Console.ResetColor(); + } + else if (content is ErrorContent errorContent) + { + Console.ForegroundColor = ConsoleColor.Red; + Console.WriteLine($"\n[Error: {errorContent.Message}]"); + Console.ResetColor(); + } + } + } + + Console.ForegroundColor = ConsoleColor.Green; + Console.WriteLine($"\n[Run Finished - Thread: {threadId}]"); + Console.ResetColor(); + } +} +catch (Exception ex) +{ + Console.WriteLine($"\nAn error occurred: {ex.Message}"); +} diff --git a/dotnet/samples/GettingStarted/AGUI/Step01_GettingStarted/Server/Program.cs b/dotnet/samples/GettingStarted/AGUI/Step01_GettingStarted/Server/Program.cs new file mode 100644 index 0000000000..1bfb9a97aa --- /dev/null +++ b/dotnet/samples/GettingStarted/AGUI/Step01_GettingStarted/Server/Program.cs @@ -0,0 +1,34 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Azure.AI.OpenAI; +using Azure.Identity; +using Microsoft.Agents.AI; +using Microsoft.Agents.AI.Hosting.AGUI.AspNetCore; +using Microsoft.Extensions.AI; +using OpenAI.Chat; + +WebApplicationBuilder builder = WebApplication.CreateBuilder(args); +builder.Services.AddHttpClient().AddLogging(); +builder.Services.AddAGUI(); + +WebApplication app = builder.Build(); + +string endpoint = builder.Configuration["AZURE_OPENAI_ENDPOINT"] + ?? throw new InvalidOperationException("AZURE_OPENAI_ENDPOINT is not set."); +string deploymentName = builder.Configuration["AZURE_OPENAI_DEPLOYMENT_NAME"] + ?? throw new InvalidOperationException("AZURE_OPENAI_DEPLOYMENT_NAME is not set."); + +// Create the AI agent +ChatClient chatClient = new AzureOpenAIClient( + new Uri(endpoint), + new DefaultAzureCredential()) + .GetChatClient(deploymentName); + +AIAgent agent = chatClient.AsIChatClient().CreateAIAgent( + name: "AGUIAssistant", + instructions: "You are a helpful assistant."); + +// Map the AG-UI agent endpoint +app.MapAGUI("/", agent); + +await app.RunAsync(); diff --git a/dotnet/samples/GettingStarted/AGUI/Step01_GettingStarted/Server/Properties/launchSettings.json b/dotnet/samples/GettingStarted/AGUI/Step01_GettingStarted/Server/Properties/launchSettings.json new file mode 100644 index 0000000000..2bac1b9426 --- /dev/null +++ b/dotnet/samples/GettingStarted/AGUI/Step01_GettingStarted/Server/Properties/launchSettings.json @@ -0,0 +1,23 @@ +{ + "$schema": "https://json.schemastore.org/launchsettings.json", + "profiles": { + "http": { + "commandName": "Project", + "dotnetRunMessages": true, + "launchBrowser": true, + "applicationUrl": "http://localhost:5253", + "environmentVariables": { + "ASPNETCORE_ENVIRONMENT": "Development" + } + }, + "https": { + "commandName": "Project", + "dotnetRunMessages": true, + "launchBrowser": true, + "applicationUrl": "https://localhost:7047;http://localhost:5253", + "environmentVariables": { + "ASPNETCORE_ENVIRONMENT": "Development" + } + } + } +} diff --git a/dotnet/samples/GettingStarted/AGUI/Step01_GettingStarted/Server/Server.csproj b/dotnet/samples/GettingStarted/AGUI/Step01_GettingStarted/Server/Server.csproj new file mode 100644 index 0000000000..b1e7fe33cf --- /dev/null +++ b/dotnet/samples/GettingStarted/AGUI/Step01_GettingStarted/Server/Server.csproj @@ -0,0 +1,21 @@ + + + + Exe + net10.0 + enable + enable + + + + + + + + + + + + + + diff --git a/dotnet/samples/GettingStarted/AGUI/Step01_GettingStarted/Server/appsettings.Development.json b/dotnet/samples/GettingStarted/AGUI/Step01_GettingStarted/Server/appsettings.Development.json new file mode 100644 index 0000000000..0c208ae918 --- /dev/null +++ b/dotnet/samples/GettingStarted/AGUI/Step01_GettingStarted/Server/appsettings.Development.json @@ -0,0 +1,8 @@ +{ + "Logging": { + "LogLevel": { + "Default": "Information", + "Microsoft.AspNetCore": "Warning" + } + } +} diff --git a/dotnet/samples/GettingStarted/AGUI/Step01_GettingStarted/Server/appsettings.json b/dotnet/samples/GettingStarted/AGUI/Step01_GettingStarted/Server/appsettings.json new file mode 100644 index 0000000000..10f68b8c8b --- /dev/null +++ b/dotnet/samples/GettingStarted/AGUI/Step01_GettingStarted/Server/appsettings.json @@ -0,0 +1,9 @@ +{ + "Logging": { + "LogLevel": { + "Default": "Information", + "Microsoft.AspNetCore": "Warning" + } + }, + "AllowedHosts": "*" +} diff --git a/dotnet/samples/GettingStarted/AGUI/Step02_BackendTools/Client/Client.csproj b/dotnet/samples/GettingStarted/AGUI/Step02_BackendTools/Client/Client.csproj new file mode 100644 index 0000000000..a76a2b37ef --- /dev/null +++ b/dotnet/samples/GettingStarted/AGUI/Step02_BackendTools/Client/Client.csproj @@ -0,0 +1,15 @@ + + + + Exe + net10.0 + enable + enable + + + + + + + + diff --git a/dotnet/samples/GettingStarted/AGUI/Step02_BackendTools/Client/Program.cs b/dotnet/samples/GettingStarted/AGUI/Step02_BackendTools/Client/Program.cs new file mode 100644 index 0000000000..1919a9565f --- /dev/null +++ b/dotnet/samples/GettingStarted/AGUI/Step02_BackendTools/Client/Program.cs @@ -0,0 +1,126 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Microsoft.Agents.AI; +using Microsoft.Agents.AI.AGUI; +using Microsoft.Extensions.AI; + +string serverUrl = Environment.GetEnvironmentVariable("AGUI_SERVER_URL") ?? "http://localhost:8888"; + +Console.WriteLine($"Connecting to AG-UI server at: {serverUrl}\n"); + +// Create the AG-UI client agent +using HttpClient httpClient = new() +{ + Timeout = TimeSpan.FromSeconds(60) +}; + +AGUIChatClient chatClient = new(httpClient, serverUrl); + +AIAgent agent = chatClient.CreateAIAgent( + name: "agui-client", + description: "AG-UI Client Agent"); + +AgentThread thread = agent.GetNewThread(); +List messages = +[ + new(ChatRole.System, "You are a helpful assistant.") +]; + +try +{ + while (true) + { + // Get user input + Console.Write("\nUser (:q or quit to exit): "); + string? message = Console.ReadLine(); + + if (string.IsNullOrWhiteSpace(message)) + { + Console.WriteLine("Request cannot be empty."); + continue; + } + + if (message is ":q" or "quit") + { + break; + } + + messages.Add(new ChatMessage(ChatRole.User, message)); + + // Stream the response + bool isFirstUpdate = true; + string? threadId = null; + + await foreach (AgentRunResponseUpdate update in agent.RunStreamingAsync(messages, thread)) + { + ChatResponseUpdate chatUpdate = update.AsChatResponseUpdate(); + + // First update indicates run started + if (isFirstUpdate) + { + threadId = chatUpdate.ConversationId; + Console.ForegroundColor = ConsoleColor.Yellow; + Console.WriteLine($"\n[Run Started - Thread: {chatUpdate.ConversationId}, Run: {chatUpdate.ResponseId}]"); + Console.ResetColor(); + isFirstUpdate = false; + } + + // Display streaming content + foreach (AIContent content in update.Contents) + { + switch (content) + { + case TextContent textContent: + Console.ForegroundColor = ConsoleColor.Cyan; + Console.Write(textContent.Text); + Console.ResetColor(); + break; + + case FunctionCallContent functionCallContent: + Console.ForegroundColor = ConsoleColor.Green; + Console.WriteLine($"\n[Function Call - Name: {functionCallContent.Name}]"); + + // Display individual parameters + if (functionCallContent.Arguments != null) + { + foreach (var kvp in functionCallContent.Arguments) + { + Console.WriteLine($" Parameter: {kvp.Key} = {kvp.Value}"); + } + } + Console.ResetColor(); + break; + + case FunctionResultContent functionResultContent: + Console.ForegroundColor = ConsoleColor.Magenta; + Console.WriteLine($"\n[Function Result - CallId: {functionResultContent.CallId}]"); + + if (functionResultContent.Exception != null) + { + Console.WriteLine($" Exception: {functionResultContent.Exception}"); + } + else + { + Console.WriteLine($" Result: {functionResultContent.Result}"); + } + Console.ResetColor(); + break; + + case ErrorContent errorContent: + Console.ForegroundColor = ConsoleColor.Red; + Console.WriteLine($"\n[Error: {errorContent.Message}]"); + Console.ResetColor(); + break; + } + } + } + + Console.ForegroundColor = ConsoleColor.Green; + Console.WriteLine($"\n[Run Finished - Thread: {threadId}]"); + Console.ResetColor(); + } +} +catch (Exception ex) +{ + Console.WriteLine($"\nAn error occurred: {ex.Message}"); +} diff --git a/dotnet/samples/GettingStarted/AGUI/Step02_BackendTools/Server/Program.cs b/dotnet/samples/GettingStarted/AGUI/Step02_BackendTools/Server/Program.cs new file mode 100644 index 0000000000..2867721d02 --- /dev/null +++ b/dotnet/samples/GettingStarted/AGUI/Step02_BackendTools/Server/Program.cs @@ -0,0 +1,117 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.ComponentModel; +using System.Text.Json.Serialization; +using Azure.AI.OpenAI; +using Azure.Identity; +using Microsoft.Agents.AI; +using Microsoft.Agents.AI.Hosting.AGUI.AspNetCore; +using Microsoft.Extensions.AI; +using Microsoft.Extensions.Options; +using OpenAI.Chat; + +WebApplicationBuilder builder = WebApplication.CreateBuilder(args); +builder.Services.AddHttpClient().AddLogging(); +builder.Services.ConfigureHttpJsonOptions(options => + options.SerializerOptions.TypeInfoResolverChain.Add(SampleJsonSerializerContext.Default)); +builder.Services.AddAGUI(); + +WebApplication app = builder.Build(); + +string endpoint = builder.Configuration["AZURE_OPENAI_ENDPOINT"] + ?? throw new InvalidOperationException("AZURE_OPENAI_ENDPOINT is not set."); +string deploymentName = builder.Configuration["AZURE_OPENAI_DEPLOYMENT_NAME"] + ?? throw new InvalidOperationException("AZURE_OPENAI_DEPLOYMENT_NAME is not set."); + +// Define the function tool +[Description("Search for restaurants in a location.")] +static RestaurantSearchResponse SearchRestaurants( + [Description("The restaurant search request")] RestaurantSearchRequest request) +{ + // Simulated restaurant data + string cuisine = request.Cuisine == "any" ? "Italian" : request.Cuisine; + + return new RestaurantSearchResponse + { + Location = request.Location, + Cuisine = request.Cuisine, + Results = + [ + new RestaurantInfo + { + Name = "The Golden Fork", + Cuisine = cuisine, + Rating = 4.5, + Address = $"123 Main St, {request.Location}" + }, + new RestaurantInfo + { + Name = "Spice Haven", + Cuisine = cuisine == "Italian" ? "Indian" : cuisine, + Rating = 4.7, + Address = $"456 Oak Ave, {request.Location}" + }, + new RestaurantInfo + { + Name = "Green Leaf", + Cuisine = "Vegetarian", + Rating = 4.3, + Address = $"789 Elm Rd, {request.Location}" + } + ] + }; +} + +// Get JsonSerializerOptions from the configured HTTP JSON options +Microsoft.AspNetCore.Http.Json.JsonOptions jsonOptions = app.Services.GetRequiredService>().Value; + +// Create tool with serializer options +AITool[] tools = +[ + AIFunctionFactory.Create( + SearchRestaurants, + serializerOptions: jsonOptions.SerializerOptions) +]; + +// Create the AI agent with tools +ChatClient chatClient = new AzureOpenAIClient( + new Uri(endpoint), + new DefaultAzureCredential()) + .GetChatClient(deploymentName); + +ChatClientAgent agent = chatClient.AsIChatClient().CreateAIAgent( + name: "AGUIAssistant", + instructions: "You are a helpful assistant with access to restaurant information.", + tools: tools); + +// Map the AG-UI agent endpoint +app.MapAGUI("/", agent); + +await app.RunAsync(); + +// Define request/response types for the tool +internal sealed class RestaurantSearchRequest +{ + public string Location { get; set; } = string.Empty; + public string Cuisine { get; set; } = "any"; +} + +internal sealed class RestaurantSearchResponse +{ + public string Location { get; set; } = string.Empty; + public string Cuisine { get; set; } = string.Empty; + public RestaurantInfo[] Results { get; set; } = []; +} + +internal sealed class RestaurantInfo +{ + public string Name { get; set; } = string.Empty; + public string Cuisine { get; set; } = string.Empty; + public double Rating { get; set; } + public string Address { get; set; } = string.Empty; +} + +// JSON serialization context for source generation +[JsonSerializable(typeof(RestaurantSearchRequest))] +[JsonSerializable(typeof(RestaurantSearchResponse))] +internal sealed partial class SampleJsonSerializerContext : JsonSerializerContext; diff --git a/dotnet/samples/GettingStarted/AGUI/Step02_BackendTools/Server/Properties/launchSettings.json b/dotnet/samples/GettingStarted/AGUI/Step02_BackendTools/Server/Properties/launchSettings.json new file mode 100644 index 0000000000..2bac1b9426 --- /dev/null +++ b/dotnet/samples/GettingStarted/AGUI/Step02_BackendTools/Server/Properties/launchSettings.json @@ -0,0 +1,23 @@ +{ + "$schema": "https://json.schemastore.org/launchsettings.json", + "profiles": { + "http": { + "commandName": "Project", + "dotnetRunMessages": true, + "launchBrowser": true, + "applicationUrl": "http://localhost:5253", + "environmentVariables": { + "ASPNETCORE_ENVIRONMENT": "Development" + } + }, + "https": { + "commandName": "Project", + "dotnetRunMessages": true, + "launchBrowser": true, + "applicationUrl": "https://localhost:7047;http://localhost:5253", + "environmentVariables": { + "ASPNETCORE_ENVIRONMENT": "Development" + } + } + } +} diff --git a/dotnet/samples/GettingStarted/AGUI/Step02_BackendTools/Server/Server.csproj b/dotnet/samples/GettingStarted/AGUI/Step02_BackendTools/Server/Server.csproj new file mode 100644 index 0000000000..b1e7fe33cf --- /dev/null +++ b/dotnet/samples/GettingStarted/AGUI/Step02_BackendTools/Server/Server.csproj @@ -0,0 +1,21 @@ + + + + Exe + net10.0 + enable + enable + + + + + + + + + + + + + + diff --git a/dotnet/samples/GettingStarted/AGUI/Step02_BackendTools/Server/appsettings.Development.json b/dotnet/samples/GettingStarted/AGUI/Step02_BackendTools/Server/appsettings.Development.json new file mode 100644 index 0000000000..0c208ae918 --- /dev/null +++ b/dotnet/samples/GettingStarted/AGUI/Step02_BackendTools/Server/appsettings.Development.json @@ -0,0 +1,8 @@ +{ + "Logging": { + "LogLevel": { + "Default": "Information", + "Microsoft.AspNetCore": "Warning" + } + } +} diff --git a/dotnet/samples/GettingStarted/AGUI/Step02_BackendTools/Server/appsettings.json b/dotnet/samples/GettingStarted/AGUI/Step02_BackendTools/Server/appsettings.json new file mode 100644 index 0000000000..10f68b8c8b --- /dev/null +++ b/dotnet/samples/GettingStarted/AGUI/Step02_BackendTools/Server/appsettings.json @@ -0,0 +1,9 @@ +{ + "Logging": { + "LogLevel": { + "Default": "Information", + "Microsoft.AspNetCore": "Warning" + } + }, + "AllowedHosts": "*" +} diff --git a/dotnet/samples/GettingStarted/AGUI/Step03_FrontendTools/Client/Client.csproj b/dotnet/samples/GettingStarted/AGUI/Step03_FrontendTools/Client/Client.csproj new file mode 100644 index 0000000000..a76a2b37ef --- /dev/null +++ b/dotnet/samples/GettingStarted/AGUI/Step03_FrontendTools/Client/Client.csproj @@ -0,0 +1,15 @@ + + + + Exe + net10.0 + enable + enable + + + + + + + + diff --git a/dotnet/samples/GettingStarted/AGUI/Step03_FrontendTools/Client/Program.cs b/dotnet/samples/GettingStarted/AGUI/Step03_FrontendTools/Client/Program.cs new file mode 100644 index 0000000000..d295ed7116 --- /dev/null +++ b/dotnet/samples/GettingStarted/AGUI/Step03_FrontendTools/Client/Program.cs @@ -0,0 +1,119 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.ComponentModel; +using Microsoft.Agents.AI; +using Microsoft.Agents.AI.AGUI; +using Microsoft.Extensions.AI; + +string serverUrl = Environment.GetEnvironmentVariable("AGUI_SERVER_URL") ?? "http://localhost:8888"; + +Console.WriteLine($"Connecting to AG-UI server at: {serverUrl}\n"); + +// Define a frontend function tool +[Description("Get the user's current location from GPS.")] +static string GetUserLocation() +{ + // Access client-side GPS + return "Amsterdam, Netherlands (52.37°N, 4.90°E)"; +} + +// Create frontend tools +AITool[] frontendTools = [AIFunctionFactory.Create(GetUserLocation)]; + +// Create the AG-UI client agent with tools +using HttpClient httpClient = new() +{ + Timeout = TimeSpan.FromSeconds(60) +}; + +AGUIChatClient chatClient = new(httpClient, serverUrl); + +AIAgent agent = chatClient.CreateAIAgent( + name: "agui-client", + description: "AG-UI Client Agent", + tools: frontendTools); + +AgentThread thread = agent.GetNewThread(); +List messages = +[ + new(ChatRole.System, "You are a helpful assistant.") +]; + +try +{ + while (true) + { + // Get user input + Console.Write("\nUser (:q or quit to exit): "); + string? message = Console.ReadLine(); + + if (string.IsNullOrWhiteSpace(message)) + { + Console.WriteLine("Request cannot be empty."); + continue; + } + + if (message is ":q" or "quit") + { + break; + } + + messages.Add(new ChatMessage(ChatRole.User, message)); + + // Stream the response + bool isFirstUpdate = true; + string? threadId = null; + + await foreach (AgentRunResponseUpdate update in agent.RunStreamingAsync(messages, thread)) + { + ChatResponseUpdate chatUpdate = update.AsChatResponseUpdate(); + + // First update indicates run started + if (isFirstUpdate) + { + threadId = chatUpdate.ConversationId; + Console.ForegroundColor = ConsoleColor.Yellow; + Console.WriteLine($"\n[Run Started - Thread: {chatUpdate.ConversationId}, Run: {chatUpdate.ResponseId}]"); + Console.ResetColor(); + isFirstUpdate = false; + } + + // Display streaming content + foreach (AIContent content in update.Contents) + { + if (content is TextContent textContent) + { + Console.ForegroundColor = ConsoleColor.Cyan; + Console.Write(textContent.Text); + Console.ResetColor(); + } + else if (content is FunctionCallContent functionCallContent) + { + Console.ForegroundColor = ConsoleColor.Green; + Console.WriteLine($"\n[Client Tool Call - Name: {functionCallContent.Name}]"); + Console.ResetColor(); + } + else if (content is FunctionResultContent functionResultContent) + { + Console.ForegroundColor = ConsoleColor.Magenta; + Console.WriteLine($"[Client Tool Result: {functionResultContent.Result}]"); + Console.ResetColor(); + } + else if (content is ErrorContent errorContent) + { + Console.ForegroundColor = ConsoleColor.Red; + Console.WriteLine($"\n[Error: {errorContent.Message}]"); + Console.ResetColor(); + } + } + } + + Console.ForegroundColor = ConsoleColor.Green; + Console.WriteLine($"\n[Run Finished - Thread: {threadId}]"); + Console.ResetColor(); + } +} +catch (Exception ex) +{ + Console.WriteLine($"\nAn error occurred: {ex.Message}"); +} diff --git a/dotnet/samples/GettingStarted/AGUI/Step03_FrontendTools/Server/Program.cs b/dotnet/samples/GettingStarted/AGUI/Step03_FrontendTools/Server/Program.cs new file mode 100644 index 0000000000..1bfb9a97aa --- /dev/null +++ b/dotnet/samples/GettingStarted/AGUI/Step03_FrontendTools/Server/Program.cs @@ -0,0 +1,34 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Azure.AI.OpenAI; +using Azure.Identity; +using Microsoft.Agents.AI; +using Microsoft.Agents.AI.Hosting.AGUI.AspNetCore; +using Microsoft.Extensions.AI; +using OpenAI.Chat; + +WebApplicationBuilder builder = WebApplication.CreateBuilder(args); +builder.Services.AddHttpClient().AddLogging(); +builder.Services.AddAGUI(); + +WebApplication app = builder.Build(); + +string endpoint = builder.Configuration["AZURE_OPENAI_ENDPOINT"] + ?? throw new InvalidOperationException("AZURE_OPENAI_ENDPOINT is not set."); +string deploymentName = builder.Configuration["AZURE_OPENAI_DEPLOYMENT_NAME"] + ?? throw new InvalidOperationException("AZURE_OPENAI_DEPLOYMENT_NAME is not set."); + +// Create the AI agent +ChatClient chatClient = new AzureOpenAIClient( + new Uri(endpoint), + new DefaultAzureCredential()) + .GetChatClient(deploymentName); + +AIAgent agent = chatClient.AsIChatClient().CreateAIAgent( + name: "AGUIAssistant", + instructions: "You are a helpful assistant."); + +// Map the AG-UI agent endpoint +app.MapAGUI("/", agent); + +await app.RunAsync(); diff --git a/dotnet/samples/GettingStarted/AGUI/Step03_FrontendTools/Server/Properties/launchSettings.json b/dotnet/samples/GettingStarted/AGUI/Step03_FrontendTools/Server/Properties/launchSettings.json new file mode 100644 index 0000000000..2bac1b9426 --- /dev/null +++ b/dotnet/samples/GettingStarted/AGUI/Step03_FrontendTools/Server/Properties/launchSettings.json @@ -0,0 +1,23 @@ +{ + "$schema": "https://json.schemastore.org/launchsettings.json", + "profiles": { + "http": { + "commandName": "Project", + "dotnetRunMessages": true, + "launchBrowser": true, + "applicationUrl": "http://localhost:5253", + "environmentVariables": { + "ASPNETCORE_ENVIRONMENT": "Development" + } + }, + "https": { + "commandName": "Project", + "dotnetRunMessages": true, + "launchBrowser": true, + "applicationUrl": "https://localhost:7047;http://localhost:5253", + "environmentVariables": { + "ASPNETCORE_ENVIRONMENT": "Development" + } + } + } +} diff --git a/dotnet/samples/GettingStarted/AGUI/Step03_FrontendTools/Server/Server.csproj b/dotnet/samples/GettingStarted/AGUI/Step03_FrontendTools/Server/Server.csproj new file mode 100644 index 0000000000..b1e7fe33cf --- /dev/null +++ b/dotnet/samples/GettingStarted/AGUI/Step03_FrontendTools/Server/Server.csproj @@ -0,0 +1,21 @@ + + + + Exe + net10.0 + enable + enable + + + + + + + + + + + + + + diff --git a/dotnet/samples/GettingStarted/AGUI/Step03_FrontendTools/Server/appsettings.Development.json b/dotnet/samples/GettingStarted/AGUI/Step03_FrontendTools/Server/appsettings.Development.json new file mode 100644 index 0000000000..0c208ae918 --- /dev/null +++ b/dotnet/samples/GettingStarted/AGUI/Step03_FrontendTools/Server/appsettings.Development.json @@ -0,0 +1,8 @@ +{ + "Logging": { + "LogLevel": { + "Default": "Information", + "Microsoft.AspNetCore": "Warning" + } + } +} diff --git a/dotnet/samples/GettingStarted/AGUI/Step03_FrontendTools/Server/appsettings.json b/dotnet/samples/GettingStarted/AGUI/Step03_FrontendTools/Server/appsettings.json new file mode 100644 index 0000000000..10f68b8c8b --- /dev/null +++ b/dotnet/samples/GettingStarted/AGUI/Step03_FrontendTools/Server/appsettings.json @@ -0,0 +1,9 @@ +{ + "Logging": { + "LogLevel": { + "Default": "Information", + "Microsoft.AspNetCore": "Warning" + } + }, + "AllowedHosts": "*" +} diff --git a/dotnet/samples/GettingStarted/AGUI/Step04_HumanInLoop/Client/Client.csproj b/dotnet/samples/GettingStarted/AGUI/Step04_HumanInLoop/Client/Client.csproj new file mode 100644 index 0000000000..a76a2b37ef --- /dev/null +++ b/dotnet/samples/GettingStarted/AGUI/Step04_HumanInLoop/Client/Client.csproj @@ -0,0 +1,15 @@ + + + + Exe + net10.0 + enable + enable + + + + + + + + diff --git a/dotnet/samples/GettingStarted/AGUI/Step04_HumanInLoop/Client/Program.cs b/dotnet/samples/GettingStarted/AGUI/Step04_HumanInLoop/Client/Program.cs new file mode 100644 index 0000000000..656989458d --- /dev/null +++ b/dotnet/samples/GettingStarted/AGUI/Step04_HumanInLoop/Client/Program.cs @@ -0,0 +1,152 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json; +using Microsoft.Agents.AI; +using Microsoft.Agents.AI.AGUI; +using Microsoft.Extensions.AI; + +string serverUrl = Environment.GetEnvironmentVariable("AGUI_SERVER_URL") ?? "http://localhost:5100"; + +// Connect to the AG-UI server +using HttpClient httpClient = new() +{ + Timeout = TimeSpan.FromSeconds(60) +}; + +AGUIChatClient chatClient = new(httpClient, serverUrl); + +// Create agent +ChatClientAgent baseAgent = chatClient.CreateAIAgent( + name: "AGUIAssistant", + instructions: "You are a helpful assistant."); + +// Use default JSON serializer options +JsonSerializerOptions jsonSerializerOptions = JsonSerializerOptions.Default; + +// Wrap the agent with ServerFunctionApprovalClientAgent +ServerFunctionApprovalClientAgent agent = new(baseAgent, jsonSerializerOptions); + +List messages = []; +AgentThread? thread = null; + +Console.ForegroundColor = ConsoleColor.White; +Console.WriteLine("Ask a question (or type 'exit' to quit):"); +Console.ResetColor(); + +string? input; +while ((input = Console.ReadLine()) != null && !input.Equals("exit", StringComparison.OrdinalIgnoreCase)) +{ + if (string.IsNullOrWhiteSpace(input)) + { + continue; + } + + messages.Add(new ChatMessage(ChatRole.User, input)); + Console.WriteLine(); + +#pragma warning disable MEAI001 + List approvalResponses = []; + + do + { + approvalResponses.Clear(); + + List chatResponseUpdates = []; + await foreach (AgentRunResponseUpdate update in agent.RunStreamingAsync(messages, thread, cancellationToken: default)) + { + chatResponseUpdates.Add(update); + foreach (AIContent content in update.Contents) + { + switch (content) + { + case FunctionApprovalRequestContent approvalRequest: + DisplayApprovalRequest(approvalRequest); + + Console.Write($"\nApprove '{approvalRequest.FunctionCall.Name}'? (yes/no): "); + string? userInput = Console.ReadLine(); + bool approved = userInput?.ToUpperInvariant() is "YES" or "Y"; + + FunctionApprovalResponseContent approvalResponse = approvalRequest.CreateResponse(approved); + + if (approvalRequest.AdditionalProperties != null) + { + approvalResponse.AdditionalProperties = new AdditionalPropertiesDictionary(); + foreach (var kvp in approvalRequest.AdditionalProperties) + { + approvalResponse.AdditionalProperties[kvp.Key] = kvp.Value; + } + } + + approvalResponses.Add(approvalResponse); + break; + + case TextContent textContent: + Console.ForegroundColor = ConsoleColor.Cyan; + Console.Write(textContent.Text); + Console.ResetColor(); + break; + + case FunctionCallContent functionCall: + Console.ForegroundColor = ConsoleColor.Green; + Console.WriteLine($"[Tool Call - Name: {functionCall.Name}]"); + if (functionCall.Arguments is { } arguments) + { + Console.WriteLine($" Parameters: {JsonSerializer.Serialize(arguments)}"); + } + Console.ResetColor(); + break; + + case FunctionResultContent functionResult: + Console.ForegroundColor = ConsoleColor.Magenta; + Console.WriteLine($"[Tool Result: {functionResult.Result}]"); + Console.ResetColor(); + break; + + case ErrorContent error: + Console.ForegroundColor = ConsoleColor.Red; + Console.WriteLine($"[Error: {error.Message}]"); + Console.ResetColor(); + break; + } + } + } + + AgentRunResponse response = chatResponseUpdates.ToAgentRunResponse(); + messages.AddRange(response.Messages); + foreach (AIContent approvalResponse in approvalResponses) + { + messages.Add(new ChatMessage(ChatRole.Tool, [approvalResponse])); + } + } + while (approvalResponses.Count > 0); +#pragma warning restore MEAI001 + + Console.WriteLine("\n"); + Console.ForegroundColor = ConsoleColor.White; + Console.WriteLine("Ask another question (or type 'exit' to quit):"); + Console.ResetColor(); +} + +#pragma warning disable MEAI001 +static void DisplayApprovalRequest(FunctionApprovalRequestContent approvalRequest) +{ + Console.ForegroundColor = ConsoleColor.Yellow; + Console.WriteLine(); + Console.WriteLine("============================================================"); + Console.WriteLine("APPROVAL REQUIRED"); + Console.WriteLine("============================================================"); + Console.WriteLine($"Function: {approvalRequest.FunctionCall.Name}"); + + if (approvalRequest.FunctionCall.Arguments != null) + { + Console.WriteLine("Arguments:"); + foreach (var arg in approvalRequest.FunctionCall.Arguments) + { + Console.WriteLine($" {arg.Key} = {arg.Value}"); + } + } + + Console.WriteLine("============================================================"); + Console.ResetColor(); +} +#pragma warning restore MEAI001 diff --git a/dotnet/samples/GettingStarted/AGUI/Step04_HumanInLoop/Client/ServerFunctionApprovalClientAgent.cs b/dotnet/samples/GettingStarted/AGUI/Step04_HumanInLoop/Client/ServerFunctionApprovalClientAgent.cs new file mode 100644 index 0000000000..41538085db --- /dev/null +++ b/dotnet/samples/GettingStarted/AGUI/Step04_HumanInLoop/Client/ServerFunctionApprovalClientAgent.cs @@ -0,0 +1,265 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Runtime.CompilerServices; +using System.Text.Json; +using System.Text.Json.Serialization; +using Microsoft.Agents.AI; +using Microsoft.Extensions.AI; +using ServerFunctionApproval; + +/// +/// A delegating agent that handles server function approval requests and responses. +/// Transforms between FunctionApprovalRequestContent/FunctionApprovalResponseContent +/// and the server's request_approval tool call pattern. +/// +internal sealed class ServerFunctionApprovalClientAgent : DelegatingAIAgent +{ + private readonly JsonSerializerOptions _jsonSerializerOptions; + + public ServerFunctionApprovalClientAgent(AIAgent innerAgent, JsonSerializerOptions jsonSerializerOptions) + : base(innerAgent) + { + this._jsonSerializerOptions = jsonSerializerOptions; + } + + public override Task RunAsync( + IEnumerable messages, + AgentThread? thread = null, + AgentRunOptions? options = null, + CancellationToken cancellationToken = default) + { + return this.RunStreamingAsync(messages, thread, options, cancellationToken) + .ToAgentRunResponseAsync(cancellationToken); + } + + public override async IAsyncEnumerable RunStreamingAsync( + IEnumerable messages, + AgentThread? thread = null, + AgentRunOptions? options = null, + [EnumeratorCancellation] CancellationToken cancellationToken = default) + { + // Process and transform approval messages, creating a new message list + var processedMessages = ProcessOutgoingServerFunctionApprovals(messages.ToList(), this._jsonSerializerOptions); + + // Run the inner agent and intercept any approval requests + await foreach (var update in this.InnerAgent.RunStreamingAsync( + processedMessages, thread, options, cancellationToken).ConfigureAwait(false)) + { + yield return ProcessIncomingServerApprovalRequests(update, this._jsonSerializerOptions); + } + } + +#pragma warning disable MEAI001 // Type is for evaluation purposes only + private static FunctionResultContent ConvertApprovalResponseToToolResult(FunctionApprovalResponseContent approvalResponse, JsonSerializerOptions jsonOptions) + { + return new FunctionResultContent( + callId: approvalResponse.Id, + result: JsonSerializer.SerializeToElement( + new ApprovalResponse + { + ApprovalId = approvalResponse.Id, + Approved = approvalResponse.Approved + }, + jsonOptions)); + } + + private static List CopyMessagesUpToIndex(List messages, int index) + { + var result = new List(index); + for (int i = 0; i < index; i++) + { + result.Add(messages[i]); + } + return result; + } + + private static List CopyContentsUpToIndex(IList contents, int index) + { + var result = new List(index); + for (int i = 0; i < index; i++) + { + result.Add(contents[i]); + } + return result; + } + + private static List ProcessOutgoingServerFunctionApprovals( + List messages, + JsonSerializerOptions jsonSerializerOptions) + { + List? result = null; + + Dictionary approvalRequests = []; + for (var messageIndex = 0; messageIndex < messages.Count; messageIndex++) + { + var message = messages[messageIndex]; + List? transformedContents = null; + + // Process each content item in the message + HashSet approvalCalls = []; + for (var contentIndex = 0; contentIndex < message.Contents.Count; contentIndex++) + { + var content = message.Contents[contentIndex]; + + // Handle pending approval requests (transform to tool call) + if (content is FunctionApprovalRequestContent approvalRequest && + approvalRequest.AdditionalProperties?.TryGetValue("original_function", out var originalFunction) == true && + originalFunction is FunctionCallContent original) + { + approvalRequests[approvalRequest.Id] = approvalRequest; + transformedContents ??= CopyContentsUpToIndex(message.Contents, contentIndex); + transformedContents.Add(original); + } + // Handle pending approval responses (transform to tool result) + else if (content is FunctionApprovalResponseContent approvalResponse && + approvalRequests.TryGetValue(approvalResponse.Id, out var correspondingRequest)) + { + transformedContents ??= CopyContentsUpToIndex(message.Contents, contentIndex); + transformedContents.Add(ConvertApprovalResponseToToolResult(approvalResponse, jsonSerializerOptions)); + approvalRequests.Remove(approvalResponse.Id); + correspondingRequest.AdditionalProperties?.Remove("original_function"); + } + // Skip historical approval content + else if (content is FunctionCallContent { Name: "request_approval" } approvalCall) + { + transformedContents ??= CopyContentsUpToIndex(message.Contents, contentIndex); + approvalCalls.Add(approvalCall.CallId); + } + else if (content is FunctionResultContent functionResult && + approvalCalls.Contains(functionResult.CallId)) + { + transformedContents ??= CopyContentsUpToIndex(message.Contents, contentIndex); + approvalCalls.Remove(functionResult.CallId); + } + else if (transformedContents != null) + { + transformedContents.Add(content); + } + } + + if (transformedContents?.Count == 0) + { + continue; + } + else if (transformedContents != null) + { + // We made changes to contents, so use transformedContents + var newMessage = new ChatMessage(message.Role, transformedContents) + { + AuthorName = message.AuthorName, + MessageId = message.MessageId, + CreatedAt = message.CreatedAt, + RawRepresentation = message.RawRepresentation, + AdditionalProperties = message.AdditionalProperties + }; + result ??= CopyMessagesUpToIndex(messages, messageIndex); + result.Add(newMessage); + } + else if (result != null) + { + // We're already copying messages, so copy this unchanged message too + result.Add(message); + } + // If result is null, we haven't made any changes yet, so keep processing + } + + return result ?? messages; + } + + private static AgentRunResponseUpdate ProcessIncomingServerApprovalRequests( + AgentRunResponseUpdate update, + JsonSerializerOptions jsonSerializerOptions) + { + IList? updatedContents = null; + for (var i = 0; i < update.Contents.Count; i++) + { + var content = update.Contents[i]; + if (content is FunctionCallContent { Name: "request_approval" } request) + { + updatedContents ??= [.. update.Contents]; + + // Serialize the function arguments as JsonElement + ApprovalRequest? approvalRequest; + if (request.Arguments?.TryGetValue("request", out var reqObj) == true && + reqObj is JsonElement je) + { + approvalRequest = (ApprovalRequest?)je.Deserialize(jsonSerializerOptions.GetTypeInfo(typeof(ApprovalRequest))); + } + else + { + approvalRequest = null; + } + + if (approvalRequest == null) + { + throw new InvalidOperationException("Failed to deserialize approval request."); + } + + var functionCallArgs = (Dictionary?)approvalRequest.FunctionArguments? + .Deserialize(jsonSerializerOptions.GetTypeInfo(typeof(Dictionary))); + + var approvalRequestContent = new FunctionApprovalRequestContent( + id: approvalRequest.ApprovalId, + new FunctionCallContent( + callId: approvalRequest.ApprovalId, + name: approvalRequest.FunctionName, + arguments: functionCallArgs)); + + approvalRequestContent.AdditionalProperties ??= []; + approvalRequestContent.AdditionalProperties["original_function"] = content; + + updatedContents[i] = approvalRequestContent; + } + } + + if (updatedContents is not null) + { + var chatUpdate = update.AsChatResponseUpdate(); + return new AgentRunResponseUpdate(new ChatResponseUpdate() + { + Role = chatUpdate.Role, + Contents = updatedContents, + MessageId = chatUpdate.MessageId, + AuthorName = chatUpdate.AuthorName, + CreatedAt = chatUpdate.CreatedAt, + RawRepresentation = chatUpdate.RawRepresentation, + ResponseId = chatUpdate.ResponseId, + AdditionalProperties = chatUpdate.AdditionalProperties + }) + { + AgentId = update.AgentId, + ContinuationToken = update.ContinuationToken, + }; + } + + return update; + } +} +#pragma warning restore MEAI001 + +namespace ServerFunctionApproval +{ + public sealed class ApprovalRequest + { + [JsonPropertyName("approval_id")] + public required string ApprovalId { get; init; } + + [JsonPropertyName("function_name")] + public required string FunctionName { get; init; } + + [JsonPropertyName("function_arguments")] + public JsonElement? FunctionArguments { get; init; } + + [JsonPropertyName("message")] + public string? Message { get; init; } + } + + public sealed class ApprovalResponse + { + [JsonPropertyName("approval_id")] + public required string ApprovalId { get; init; } + + [JsonPropertyName("approved")] + public required bool Approved { get; init; } + } +} diff --git a/dotnet/samples/GettingStarted/AGUI/Step04_HumanInLoop/Server/Program.cs b/dotnet/samples/GettingStarted/AGUI/Step04_HumanInLoop/Server/Program.cs new file mode 100644 index 0000000000..1af163435a --- /dev/null +++ b/dotnet/samples/GettingStarted/AGUI/Step04_HumanInLoop/Server/Program.cs @@ -0,0 +1,69 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.ComponentModel; +using Azure.AI.OpenAI; +using Azure.Identity; +using Microsoft.Agents.AI; +using Microsoft.Agents.AI.Hosting.AGUI.AspNetCore; +using Microsoft.AspNetCore.Http.Json; +using Microsoft.AspNetCore.HttpLogging; +using Microsoft.Extensions.AI; +using Microsoft.Extensions.Options; +using OpenAI.Chat; +using ServerFunctionApproval; + +WebApplicationBuilder builder = WebApplication.CreateBuilder(args); + +builder.Services.AddHttpLogging(logging => +{ + logging.LoggingFields = HttpLoggingFields.RequestPropertiesAndHeaders | HttpLoggingFields.RequestBody + | HttpLoggingFields.ResponsePropertiesAndHeaders | HttpLoggingFields.ResponseBody; + logging.RequestBodyLogLimit = int.MaxValue; + logging.ResponseBodyLogLimit = int.MaxValue; +}); + +builder.Services.AddHttpClient().AddLogging(); +builder.Services.ConfigureHttpJsonOptions(options => + options.SerializerOptions.TypeInfoResolverChain.Add(ApprovalJsonContext.Default)); +builder.Services.AddAGUI(); + +WebApplication app = builder.Build(); + +app.UseHttpLogging(); + +string endpoint = builder.Configuration["AZURE_OPENAI_ENDPOINT"] + ?? throw new InvalidOperationException("AZURE_OPENAI_ENDPOINT is not set."); +string deploymentName = builder.Configuration["AZURE_OPENAI_DEPLOYMENT_NAME"] + ?? throw new InvalidOperationException("AZURE_OPENAI_DEPLOYMENT_NAME is not set."); + +// Define approval-required tool +[Description("Approve the expense report.")] +static string ApproveExpenseReport(string expenseReportId) +{ + return $"Expense report {expenseReportId} approved"; +} + +// Get JsonSerializerOptions +var jsonOptions = app.Services.GetRequiredService>().Value; + +// Create approval-required tool +#pragma warning disable MEAI001 // Type is for evaluation purposes only +AITool[] tools = [new ApprovalRequiredAIFunction(AIFunctionFactory.Create(ApproveExpenseReport))]; +#pragma warning restore MEAI001 + +// Create base agent +ChatClient openAIChatClient = new AzureOpenAIClient( + new Uri(endpoint), + new DefaultAzureCredential()) + .GetChatClient(deploymentName); + +ChatClientAgent baseAgent = openAIChatClient.AsIChatClient().CreateAIAgent( + name: "AGUIAssistant", + instructions: "You are a helpful assistant in charge of approving expenses", + tools: tools); + +// Wrap with ServerFunctionApprovalAgent +var agent = new ServerFunctionApprovalAgent(baseAgent, jsonOptions.SerializerOptions); + +app.MapAGUI("/", agent); +await app.RunAsync(); diff --git a/dotnet/samples/GettingStarted/AGUI/Step04_HumanInLoop/Server/Properties/launchSettings.json b/dotnet/samples/GettingStarted/AGUI/Step04_HumanInLoop/Server/Properties/launchSettings.json new file mode 100644 index 0000000000..e75f8f51e3 --- /dev/null +++ b/dotnet/samples/GettingStarted/AGUI/Step04_HumanInLoop/Server/Properties/launchSettings.json @@ -0,0 +1,23 @@ +{ + "$schema": "https://json.schemastore.org/launchsettings.json", + "profiles": { + "http": { + "commandName": "Project", + "dotnetRunMessages": true, + "launchBrowser": true, + "applicationUrl": "http://localhost:5100", + "environmentVariables": { + "ASPNETCORE_ENVIRONMENT": "Development" + } + }, + "https": { + "commandName": "Project", + "dotnetRunMessages": true, + "launchBrowser": true, + "applicationUrl": "https://localhost:7047;http://localhost:5100", + "environmentVariables": { + "ASPNETCORE_ENVIRONMENT": "Development" + } + } + } +} diff --git a/dotnet/samples/GettingStarted/AGUI/Step04_HumanInLoop/Server/Server.csproj b/dotnet/samples/GettingStarted/AGUI/Step04_HumanInLoop/Server/Server.csproj new file mode 100644 index 0000000000..b1e7fe33cf --- /dev/null +++ b/dotnet/samples/GettingStarted/AGUI/Step04_HumanInLoop/Server/Server.csproj @@ -0,0 +1,21 @@ + + + + Exe + net10.0 + enable + enable + + + + + + + + + + + + + + diff --git a/dotnet/samples/GettingStarted/AGUI/Step04_HumanInLoop/Server/ServerFunctionApprovalServerAgent.cs b/dotnet/samples/GettingStarted/AGUI/Step04_HumanInLoop/Server/ServerFunctionApprovalServerAgent.cs new file mode 100644 index 0000000000..f515e97531 --- /dev/null +++ b/dotnet/samples/GettingStarted/AGUI/Step04_HumanInLoop/Server/ServerFunctionApprovalServerAgent.cs @@ -0,0 +1,262 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Runtime.CompilerServices; +using System.Text.Json; +using System.Text.Json.Serialization; +using Microsoft.Agents.AI; +using Microsoft.Extensions.AI; +using ServerFunctionApproval; + +/// +/// A delegating agent that handles function approval requests on the server side. +/// Transforms between FunctionApprovalRequestContent/FunctionApprovalResponseContent +/// and the request_approval tool call pattern for client communication. +/// +internal sealed class ServerFunctionApprovalAgent : DelegatingAIAgent +{ + private readonly JsonSerializerOptions _jsonSerializerOptions; + + public ServerFunctionApprovalAgent(AIAgent innerAgent, JsonSerializerOptions jsonSerializerOptions) + : base(innerAgent) + { + this._jsonSerializerOptions = jsonSerializerOptions; + } + + public override Task RunAsync( + IEnumerable messages, + AgentThread? thread = null, + AgentRunOptions? options = null, + CancellationToken cancellationToken = default) + { + return this.RunStreamingAsync(messages, thread, options, cancellationToken) + .ToAgentRunResponseAsync(cancellationToken); + } + + public override async IAsyncEnumerable RunStreamingAsync( + IEnumerable messages, + AgentThread? thread = null, + AgentRunOptions? options = null, + [EnumeratorCancellation] CancellationToken cancellationToken = default) + { + // Process and transform incoming approval responses from client, creating a new message list + var processedMessages = ProcessIncomingFunctionApprovals(messages.ToList(), this._jsonSerializerOptions); + + // Run the inner agent and intercept any approval requests + await foreach (var update in this.InnerAgent.RunStreamingAsync( + processedMessages, thread, options, cancellationToken).ConfigureAwait(false)) + { + yield return ProcessOutgoingApprovalRequests(update, this._jsonSerializerOptions); + } + } + +#pragma warning disable MEAI001 // Type is for evaluation purposes only + private static FunctionApprovalRequestContent ConvertToolCallToApprovalRequest(FunctionCallContent toolCall, JsonSerializerOptions jsonSerializerOptions) + { + if (toolCall.Name != "request_approval" || toolCall.Arguments == null) + { + throw new InvalidOperationException("Invalid request_approval tool call"); + } + + var request = toolCall.Arguments.TryGetValue("request", out var reqObj) && + reqObj is JsonElement argsElement && + argsElement.Deserialize(jsonSerializerOptions.GetTypeInfo(typeof(ApprovalRequest))) is ApprovalRequest approvalRequest && + approvalRequest != null ? approvalRequest : null; + + if (request == null) + { + throw new InvalidOperationException("Failed to deserialize approval request from tool call"); + } + + return new FunctionApprovalRequestContent( + id: request.ApprovalId, + new FunctionCallContent( + callId: request.ApprovalId, + name: request.FunctionName, + arguments: request.FunctionArguments)); + } + + private static FunctionApprovalResponseContent ConvertToolResultToApprovalResponse(FunctionResultContent result, FunctionApprovalRequestContent approval, JsonSerializerOptions jsonSerializerOptions) + { + var approvalResponse = result.Result is JsonElement je ? + (ApprovalResponse?)je.Deserialize(jsonSerializerOptions.GetTypeInfo(typeof(ApprovalResponse))) : + result.Result is string str ? + (ApprovalResponse?)JsonSerializer.Deserialize(str, jsonSerializerOptions.GetTypeInfo(typeof(ApprovalResponse))) : + result.Result as ApprovalResponse; + + if (approvalResponse == null) + { + throw new InvalidOperationException("Failed to deserialize approval response from tool result"); + } + + return approval.CreateResponse(approvalResponse.Approved); + } +#pragma warning restore MEAI001 + + private static List CopyMessagesUpToIndex(List messages, int index) + { + var result = new List(index); + for (int i = 0; i < index; i++) + { + result.Add(messages[i]); + } + return result; + } + + private static List CopyContentsUpToIndex(IList contents, int index) + { + var result = new List(index); + for (int i = 0; i < index; i++) + { + result.Add(contents[i]); + } + return result; + } + + private static List ProcessIncomingFunctionApprovals( + List messages, + JsonSerializerOptions jsonSerializerOptions) + { + List? result = null; + + // Track approval ID to original call ID mapping + _ = new Dictionary(); +#pragma warning disable MEAI001 // Type is for evaluation purposes only and is subject to change or removal in future updates. Suppress this diagnostic to proceed. + Dictionary trackedRequestApprovalToolCalls = new(); // Remote approvals + for (int messageIndex = 0; messageIndex < messages.Count; messageIndex++) + { + var message = messages[messageIndex]; + List? transformedContents = null; + for (int j = 0; j < message.Contents.Count; j++) + { + var content = message.Contents[j]; + if (content is FunctionCallContent { Name: "request_approval" } toolCall) + { + result ??= CopyMessagesUpToIndex(messages, messageIndex); + transformedContents ??= CopyContentsUpToIndex(message.Contents, j); + var approvalRequest = ConvertToolCallToApprovalRequest(toolCall, jsonSerializerOptions); + transformedContents.Add(approvalRequest); + trackedRequestApprovalToolCalls[toolCall.CallId] = approvalRequest; + result.Add(new ChatMessage(message.Role, transformedContents) + { + AuthorName = message.AuthorName, + MessageId = message.MessageId, + CreatedAt = message.CreatedAt, + RawRepresentation = message.RawRepresentation, + AdditionalProperties = message.AdditionalProperties + }); + } + else if (content is FunctionResultContent toolResult && + trackedRequestApprovalToolCalls.TryGetValue(toolResult.CallId, out var approval) == true) + { + result ??= CopyMessagesUpToIndex(messages, messageIndex); + transformedContents ??= CopyContentsUpToIndex(message.Contents, j); + var approvalResponse = ConvertToolResultToApprovalResponse(toolResult, approval, jsonSerializerOptions); + transformedContents.Add(approvalResponse); + result.Add(new ChatMessage(message.Role, transformedContents) + { + AuthorName = message.AuthorName, + MessageId = message.MessageId, + CreatedAt = message.CreatedAt, + RawRepresentation = message.RawRepresentation, + AdditionalProperties = message.AdditionalProperties + }); + } + else if (result != null) + { + result.Add(message); + } + } + } +#pragma warning restore MEAI001 + + return result ?? messages; + } + + private static AgentRunResponseUpdate ProcessOutgoingApprovalRequests( + AgentRunResponseUpdate update, + JsonSerializerOptions jsonSerializerOptions) + { + IList? updatedContents = null; + for (var i = 0; i < update.Contents.Count; i++) + { + var content = update.Contents[i]; +#pragma warning disable MEAI001 // Type is for evaluation purposes only + if (content is FunctionApprovalRequestContent request) + { + updatedContents ??= [.. update.Contents]; + var functionCall = request.FunctionCall; + var approvalId = request.Id; + + var approvalData = new ApprovalRequest + { + ApprovalId = approvalId, + FunctionName = functionCall.Name, + FunctionArguments = functionCall.Arguments, + Message = $"Approve execution of '{functionCall.Name}'?" + }; + + updatedContents[i] = new FunctionCallContent( + callId: approvalId, + name: "request_approval", + arguments: new Dictionary { ["request"] = approvalData }); + } +#pragma warning restore MEAI001 + } + + if (updatedContents is not null) + { + var chatUpdate = update.AsChatResponseUpdate(); + // Yield a tool call update that represents the approval request + return new AgentRunResponseUpdate(new ChatResponseUpdate() + { + Role = chatUpdate.Role, + Contents = updatedContents, + MessageId = chatUpdate.MessageId, + AuthorName = chatUpdate.AuthorName, + CreatedAt = chatUpdate.CreatedAt, + RawRepresentation = chatUpdate.RawRepresentation, + ResponseId = chatUpdate.ResponseId, + AdditionalProperties = chatUpdate.AdditionalProperties + }) + { + AgentId = update.AgentId, + ContinuationToken = update.ContinuationToken + }; + } + + return update; + } +} + +namespace ServerFunctionApproval +{ + // Define approval models + public sealed class ApprovalRequest + { + [JsonPropertyName("approval_id")] + public required string ApprovalId { get; init; } + + [JsonPropertyName("function_name")] + public required string FunctionName { get; init; } + + [JsonPropertyName("function_arguments")] + public IDictionary? FunctionArguments { get; init; } + + [JsonPropertyName("message")] + public string? Message { get; init; } + } + + public sealed class ApprovalResponse + { + [JsonPropertyName("approval_id")] + public required string ApprovalId { get; init; } + + [JsonPropertyName("approved")] + public required bool Approved { get; init; } + } + + [JsonSerializable(typeof(ApprovalRequest))] + [JsonSerializable(typeof(ApprovalResponse))] + [JsonSerializable(typeof(Dictionary))] + public sealed partial class ApprovalJsonContext : JsonSerializerContext; +} diff --git a/dotnet/samples/GettingStarted/AGUI/Step04_HumanInLoop/Server/appsettings.Development.json b/dotnet/samples/GettingStarted/AGUI/Step04_HumanInLoop/Server/appsettings.Development.json new file mode 100644 index 0000000000..3e805edef8 --- /dev/null +++ b/dotnet/samples/GettingStarted/AGUI/Step04_HumanInLoop/Server/appsettings.Development.json @@ -0,0 +1,9 @@ +{ + "Logging": { + "LogLevel": { + "Default": "Information", + "Microsoft.AspNetCore": "Warning", + "Microsoft.AspNetCore.HttpLogging.HttpLoggingMiddleware": "Information" + } + } +} diff --git a/dotnet/samples/GettingStarted/AGUI/Step04_HumanInLoop/Server/appsettings.json b/dotnet/samples/GettingStarted/AGUI/Step04_HumanInLoop/Server/appsettings.json new file mode 100644 index 0000000000..10f68b8c8b --- /dev/null +++ b/dotnet/samples/GettingStarted/AGUI/Step04_HumanInLoop/Server/appsettings.json @@ -0,0 +1,9 @@ +{ + "Logging": { + "LogLevel": { + "Default": "Information", + "Microsoft.AspNetCore": "Warning" + } + }, + "AllowedHosts": "*" +} diff --git a/dotnet/samples/GettingStarted/AGUI/Step05_StateManagement/Client/Client.csproj b/dotnet/samples/GettingStarted/AGUI/Step05_StateManagement/Client/Client.csproj new file mode 100644 index 0000000000..a76a2b37ef --- /dev/null +++ b/dotnet/samples/GettingStarted/AGUI/Step05_StateManagement/Client/Client.csproj @@ -0,0 +1,15 @@ + + + + Exe + net10.0 + enable + enable + + + + + + + + diff --git a/dotnet/samples/GettingStarted/AGUI/Step05_StateManagement/Client/Program.cs b/dotnet/samples/GettingStarted/AGUI/Step05_StateManagement/Client/Program.cs new file mode 100644 index 0000000000..49ffa0587d --- /dev/null +++ b/dotnet/samples/GettingStarted/AGUI/Step05_StateManagement/Client/Program.cs @@ -0,0 +1,231 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json; +using System.Text.Json.Serialization; +using Microsoft.Agents.AI; +using Microsoft.Agents.AI.AGUI; +using Microsoft.Extensions.AI; +using RecipeClient; + +string serverUrl = Environment.GetEnvironmentVariable("AGUI_SERVER_URL") ?? "http://localhost:8888"; + +Console.WriteLine($"Connecting to AG-UI server at: {serverUrl}\n"); + +// Create the AG-UI client agent +using HttpClient httpClient = new() +{ + Timeout = TimeSpan.FromSeconds(60) +}; + +AGUIChatClient chatClient = new(httpClient, serverUrl); + +AIAgent baseAgent = chatClient.CreateAIAgent( + name: "recipe-client", + description: "AG-UI Recipe Client Agent"); + +// Wrap the base agent with state management +JsonSerializerOptions jsonOptions = new(JsonSerializerDefaults.Web) +{ + TypeInfoResolver = RecipeSerializerContext.Default +}; +StatefulAgent agent = new(baseAgent, jsonOptions, new AgentState()); + +AgentThread thread = agent.GetNewThread(); +List messages = +[ + new(ChatRole.System, "You are a helpful recipe assistant.") +]; + +try +{ + while (true) + { + // Get user input + Console.Write("\nUser (:q to quit, :state to show state): "); + string? message = Console.ReadLine(); + + if (string.IsNullOrWhiteSpace(message)) + { + Console.WriteLine("Request cannot be empty."); + continue; + } + + if (message is ":q" or "quit") + { + break; + } + + if (message.Equals(":state", StringComparison.OrdinalIgnoreCase)) + { + DisplayState(agent.State.Recipe); + continue; + } + + messages.Add(new ChatMessage(ChatRole.User, message)); + + // Stream the response + bool isFirstUpdate = true; + string? threadId = null; + bool stateReceived = false; + + Console.WriteLine(); + + await foreach (AgentRunResponseUpdate update in agent.RunStreamingAsync(messages, thread)) + { + ChatResponseUpdate chatUpdate = update.AsChatResponseUpdate(); + + // First update indicates run started + if (isFirstUpdate) + { + threadId = chatUpdate.ConversationId; + Console.ForegroundColor = ConsoleColor.Yellow; + Console.WriteLine($"[Run Started - Thread: {chatUpdate.ConversationId}, Run: {chatUpdate.ResponseId}]"); + Console.ResetColor(); + isFirstUpdate = false; + } + + // Display streaming content + foreach (AIContent content in update.Contents) + { + switch (content) + { + case TextContent textContent: + Console.ForegroundColor = ConsoleColor.Cyan; + Console.Write(textContent.Text); + Console.ResetColor(); + break; + + case DataContent dataContent when dataContent.MediaType == "application/json": + // This is a state snapshot - the StatefulAgent has already updated the state + stateReceived = true; + Console.ForegroundColor = ConsoleColor.Blue; + Console.WriteLine("\n[State Snapshot Received]"); + Console.ResetColor(); + break; + + case ErrorContent errorContent: + Console.ForegroundColor = ConsoleColor.Red; + Console.WriteLine($"\n[Error: {errorContent.Message}]"); + Console.ResetColor(); + break; + } + } + } + + Console.ForegroundColor = ConsoleColor.Green; + Console.WriteLine($"\n[Run Finished - Thread: {threadId}]"); + Console.ResetColor(); + + // Display final state if received + if (stateReceived) + { + DisplayState(agent.State.Recipe); + } + } +} +catch (Exception ex) +{ + Console.WriteLine($"\nAn error occurred: {ex.Message}"); +} + +static void DisplayState(RecipeState? state) +{ + if (state == null) + { + Console.ForegroundColor = ConsoleColor.Gray; + Console.WriteLine("\n[No state available]"); + Console.ResetColor(); + return; + } + + Console.ForegroundColor = ConsoleColor.Blue; + Console.WriteLine("\n" + new string('=', 60)); + Console.WriteLine("CURRENT STATE"); + Console.WriteLine(new string('=', 60)); + Console.ResetColor(); + + if (!string.IsNullOrEmpty(state.Title)) + { + Console.WriteLine("\nRecipe:"); + Console.WriteLine($" Title: {state.Title}"); + if (!string.IsNullOrEmpty(state.Cuisine)) + { + Console.WriteLine($" Cuisine: {state.Cuisine}"); + } + + if (!string.IsNullOrEmpty(state.SkillLevel)) + { + Console.WriteLine($" Skill Level: {state.SkillLevel}"); + } + + if (state.PrepTimeMinutes > 0) + { + Console.WriteLine($" Prep Time: {state.PrepTimeMinutes} minutes"); + } + + if (state.CookTimeMinutes > 0) + { + Console.WriteLine($" Cook Time: {state.CookTimeMinutes} minutes"); + } + + if (state.Ingredients.Count > 0) + { + Console.WriteLine("\n Ingredients:"); + foreach (var ingredient in state.Ingredients) + { + Console.WriteLine($" - {ingredient}"); + } + } + + if (state.Steps.Count > 0) + { + Console.WriteLine("\n Steps:"); + for (int i = 0; i < state.Steps.Count; i++) + { + Console.WriteLine($" {i + 1}. {state.Steps[i]}"); + } + } + } + + Console.ForegroundColor = ConsoleColor.Blue; + Console.WriteLine("\n" + new string('=', 60)); + Console.ResetColor(); +} + +// State wrapper +internal sealed class AgentState +{ + [JsonPropertyName("recipe")] + public RecipeState Recipe { get; set; } = new(); +} + +// Recipe state model +internal sealed class RecipeState +{ + [JsonPropertyName("title")] + public string Title { get; set; } = string.Empty; + + [JsonPropertyName("cuisine")] + public string Cuisine { get; set; } = string.Empty; + + [JsonPropertyName("ingredients")] + public List Ingredients { get; set; } = []; + + [JsonPropertyName("steps")] + public List Steps { get; set; } = []; + + [JsonPropertyName("prep_time_minutes")] + public int PrepTimeMinutes { get; set; } + + [JsonPropertyName("cook_time_minutes")] + public int CookTimeMinutes { get; set; } + + [JsonPropertyName("skill_level")] + public string SkillLevel { get; set; } = string.Empty; +} + +// JSON serialization context +[JsonSerializable(typeof(AgentState))] +[JsonSerializable(typeof(RecipeState))] +[JsonSerializable(typeof(JsonElement))] +internal sealed partial class RecipeSerializerContext : JsonSerializerContext; diff --git a/dotnet/samples/GettingStarted/AGUI/Step05_StateManagement/Client/StatefulAgent.cs b/dotnet/samples/GettingStarted/AGUI/Step05_StateManagement/Client/StatefulAgent.cs new file mode 100644 index 0000000000..8321efaa73 --- /dev/null +++ b/dotnet/samples/GettingStarted/AGUI/Step05_StateManagement/Client/StatefulAgent.cs @@ -0,0 +1,88 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Runtime.CompilerServices; +using System.Text.Json; +using Microsoft.Agents.AI; +using Microsoft.Extensions.AI; + +namespace RecipeClient; + +/// +/// A delegating agent that manages client-side state and automatically attaches it to requests. +/// +/// The state type. +internal sealed class StatefulAgent : DelegatingAIAgent + where TState : class, new() +{ + private readonly JsonSerializerOptions _jsonSerializerOptions; + + /// + /// Gets or sets the current state. + /// + public TState State { get; set; } + + /// + /// Initializes a new instance of the class. + /// + /// The underlying agent to delegate to. + /// The JSON serializer options for state serialization. + /// The initial state. If null, a new instance will be created. + public StatefulAgent(AIAgent innerAgent, JsonSerializerOptions jsonSerializerOptions, TState? initialState = null) + : base(innerAgent) + { + this._jsonSerializerOptions = jsonSerializerOptions; + this.State = initialState ?? new TState(); + } + + /// + public override Task RunAsync( + IEnumerable messages, + AgentThread? thread = null, + AgentRunOptions? options = null, + CancellationToken cancellationToken = default) + { + return this.RunStreamingAsync(messages, thread, options, cancellationToken) + .ToAgentRunResponseAsync(cancellationToken); + } + + /// + public override async IAsyncEnumerable RunStreamingAsync( + IEnumerable messages, + AgentThread? thread = null, + AgentRunOptions? options = null, + [EnumeratorCancellation] CancellationToken cancellationToken = default) + { + // Add state to messages + List messagesWithState = [.. messages]; + + // Serialize the state using AgentState wrapper + byte[] stateBytes = JsonSerializer.SerializeToUtf8Bytes( + this.State, + this._jsonSerializerOptions.GetTypeInfo(typeof(TState))); + DataContent stateContent = new(stateBytes, "application/json"); + ChatMessage stateMessage = new(ChatRole.System, [stateContent]); + messagesWithState.Add(stateMessage); + + // Stream the response and update state when received + await foreach (AgentRunResponseUpdate update in this.InnerAgent.RunStreamingAsync(messagesWithState, thread, options, cancellationToken)) + { + // Check if this update contains a state snapshot + foreach (AIContent content in update.Contents) + { + if (content is DataContent dataContent && dataContent.MediaType == "application/json") + { + // Deserialize the state + TState? newState = JsonSerializer.Deserialize( + dataContent.Data.Span, + this._jsonSerializerOptions.GetTypeInfo(typeof(TState))) as TState; + if (newState != null) + { + this.State = newState; + } + } + } + + yield return update; + } + } +} diff --git a/dotnet/samples/GettingStarted/AGUI/Step05_StateManagement/Server/Program.cs b/dotnet/samples/GettingStarted/AGUI/Step05_StateManagement/Server/Program.cs new file mode 100644 index 0000000000..40c51887d1 --- /dev/null +++ b/dotnet/samples/GettingStarted/AGUI/Step05_StateManagement/Server/Program.cs @@ -0,0 +1,59 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Azure.AI.OpenAI; +using Azure.Identity; +using Microsoft.Agents.AI; +using Microsoft.Agents.AI.Hosting.AGUI.AspNetCore; +using Microsoft.Extensions.AI; +using Microsoft.Extensions.Options; +using OpenAI.Chat; +using RecipeAssistant; + +WebApplicationBuilder builder = WebApplication.CreateBuilder(args); +builder.Services.AddHttpClient().AddLogging(); +builder.Services.ConfigureHttpJsonOptions(options => + options.SerializerOptions.TypeInfoResolverChain.Add(RecipeSerializerContext.Default)); +builder.Services.AddAGUI(); + +// Configure to listen on port 8888 +builder.WebHost.UseUrls("http://localhost:8888"); + +WebApplication app = builder.Build(); + +string endpoint = builder.Configuration["AZURE_OPENAI_ENDPOINT"] + ?? throw new InvalidOperationException("AZURE_OPENAI_ENDPOINT is not set."); +string deploymentName = builder.Configuration["AZURE_OPENAI_DEPLOYMENT_NAME"] + ?? throw new InvalidOperationException("AZURE_OPENAI_DEPLOYMENT_NAME is not set."); + +// Get JsonSerializerOptions +var jsonOptions = app.Services.GetRequiredService>().Value; + +// Create base agent +ChatClient chatClient = new AzureOpenAIClient( + new Uri(endpoint), + new DefaultAzureCredential()) + .GetChatClient(deploymentName); + +AIAgent baseAgent = chatClient.AsIChatClient().CreateAIAgent( + name: "RecipeAgent", + instructions: """ + You are a helpful recipe assistant. When users ask you to create or suggest a recipe, + respond with a complete AgentState JSON object that includes: + - recipe.title: The recipe name + - recipe.cuisine: Type of cuisine (e.g., Italian, Mexican, Japanese) + - recipe.ingredients: Array of ingredient strings with quantities + - recipe.steps: Array of cooking instruction strings + - recipe.prep_time_minutes: Preparation time in minutes + - recipe.cook_time_minutes: Cooking time in minutes + - recipe.skill_level: One of "beginner", "intermediate", or "advanced" + + Always include all fields in the response. Be creative and helpful. + """); + +// Wrap with state management middleware +AIAgent agent = new SharedStateAgent(baseAgent, jsonOptions.SerializerOptions); + +// Map the AG-UI agent endpoint +app.MapAGUI("/", agent); + +await app.RunAsync(); diff --git a/dotnet/samples/GettingStarted/AGUI/Step05_StateManagement/Server/Properties/launchSettings.json b/dotnet/samples/GettingStarted/AGUI/Step05_StateManagement/Server/Properties/launchSettings.json new file mode 100644 index 0000000000..2bac1b9426 --- /dev/null +++ b/dotnet/samples/GettingStarted/AGUI/Step05_StateManagement/Server/Properties/launchSettings.json @@ -0,0 +1,23 @@ +{ + "$schema": "https://json.schemastore.org/launchsettings.json", + "profiles": { + "http": { + "commandName": "Project", + "dotnetRunMessages": true, + "launchBrowser": true, + "applicationUrl": "http://localhost:5253", + "environmentVariables": { + "ASPNETCORE_ENVIRONMENT": "Development" + } + }, + "https": { + "commandName": "Project", + "dotnetRunMessages": true, + "launchBrowser": true, + "applicationUrl": "https://localhost:7047;http://localhost:5253", + "environmentVariables": { + "ASPNETCORE_ENVIRONMENT": "Development" + } + } + } +} diff --git a/dotnet/samples/GettingStarted/AGUI/Step05_StateManagement/Server/RecipeModels.cs b/dotnet/samples/GettingStarted/AGUI/Step05_StateManagement/Server/RecipeModels.cs new file mode 100644 index 0000000000..fc1d8320d2 --- /dev/null +++ b/dotnet/samples/GettingStarted/AGUI/Step05_StateManagement/Server/RecipeModels.cs @@ -0,0 +1,43 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +namespace RecipeAssistant; + +// State wrapper +internal sealed class AgentState +{ + [JsonPropertyName("recipe")] + public RecipeState Recipe { get; set; } = new(); +} + +// Recipe state model +internal sealed class RecipeState +{ + [JsonPropertyName("title")] + public string Title { get; set; } = string.Empty; + + [JsonPropertyName("cuisine")] + public string Cuisine { get; set; } = string.Empty; + + [JsonPropertyName("ingredients")] + public List Ingredients { get; set; } = []; + + [JsonPropertyName("steps")] + public List Steps { get; set; } = []; + + [JsonPropertyName("prep_time_minutes")] + public int PrepTimeMinutes { get; set; } + + [JsonPropertyName("cook_time_minutes")] + public int CookTimeMinutes { get; set; } + + [JsonPropertyName("skill_level")] + public string SkillLevel { get; set; } = string.Empty; +} + +// JSON serialization context +[JsonSerializable(typeof(AgentState))] +[JsonSerializable(typeof(RecipeState))] +[JsonSerializable(typeof(System.Text.Json.JsonElement))] +internal sealed partial class RecipeSerializerContext : JsonSerializerContext; diff --git a/dotnet/samples/GettingStarted/AGUI/Step05_StateManagement/Server/Server.csproj b/dotnet/samples/GettingStarted/AGUI/Step05_StateManagement/Server/Server.csproj new file mode 100644 index 0000000000..b1e7fe33cf --- /dev/null +++ b/dotnet/samples/GettingStarted/AGUI/Step05_StateManagement/Server/Server.csproj @@ -0,0 +1,21 @@ + + + + Exe + net10.0 + enable + enable + + + + + + + + + + + + + + diff --git a/dotnet/samples/GettingStarted/AGUI/Step05_StateManagement/Server/SharedStateAgent.cs b/dotnet/samples/GettingStarted/AGUI/Step05_StateManagement/Server/SharedStateAgent.cs new file mode 100644 index 0000000000..4588c7bd60 --- /dev/null +++ b/dotnet/samples/GettingStarted/AGUI/Step05_StateManagement/Server/SharedStateAgent.cs @@ -0,0 +1,137 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Runtime.CompilerServices; +using System.Text.Json; +using Microsoft.Agents.AI; +using Microsoft.Extensions.AI; + +namespace RecipeAssistant; + +internal sealed class SharedStateAgent : DelegatingAIAgent +{ + private readonly JsonSerializerOptions _jsonSerializerOptions; + + public SharedStateAgent(AIAgent innerAgent, JsonSerializerOptions jsonSerializerOptions) + : base(innerAgent) + { + this._jsonSerializerOptions = jsonSerializerOptions; + } + + public override Task RunAsync( + IEnumerable messages, + AgentThread? thread = null, + AgentRunOptions? options = null, + CancellationToken cancellationToken = default) + { + return this.RunStreamingAsync(messages, thread, options, cancellationToken) + .ToAgentRunResponseAsync(cancellationToken); + } + + public override async IAsyncEnumerable RunStreamingAsync( + IEnumerable messages, + AgentThread? thread = null, + AgentRunOptions? options = null, + [EnumeratorCancellation] CancellationToken cancellationToken = default) + { + // Check if the client sent state in the request + if (options is not ChatClientAgentRunOptions { ChatOptions.AdditionalProperties: { } properties } chatRunOptions || + !properties.TryGetValue("ag_ui_state", out object? stateObj) || + stateObj is not JsonElement state || + state.ValueKind != JsonValueKind.Object) + { + // No state management requested, pass through to inner agent + await foreach (var update in this.InnerAgent.RunStreamingAsync(messages, thread, options, cancellationToken).ConfigureAwait(false)) + { + yield return update; + } + yield break; + } + + // Check if state has properties (not empty {}) + bool hasProperties = false; + foreach (JsonProperty _ in state.EnumerateObject()) + { + hasProperties = true; + break; + } + + if (!hasProperties) + { + // Empty state - treat as no state + await foreach (var update in this.InnerAgent.RunStreamingAsync(messages, thread, options, cancellationToken).ConfigureAwait(false)) + { + yield return update; + } + yield break; + } + + // First run: Generate structured state update + var firstRunOptions = new ChatClientAgentRunOptions + { + ChatOptions = chatRunOptions.ChatOptions.Clone(), + AllowBackgroundResponses = chatRunOptions.AllowBackgroundResponses, + ContinuationToken = chatRunOptions.ContinuationToken, + ChatClientFactory = chatRunOptions.ChatClientFactory, + }; + + // Configure JSON schema response format for structured state output + firstRunOptions.ChatOptions.ResponseFormat = ChatResponseFormat.ForJsonSchema( + schemaName: "AgentState", + schemaDescription: "A response containing a recipe with title, skill level, cooking time, ingredients, and instructions"); + + // Add current state to the conversation - state is already a JsonElement + ChatMessage stateUpdateMessage = new( + ChatRole.System, + [ + new TextContent("Here is the current state in JSON format:"), + new TextContent(JsonSerializer.Serialize(state, this._jsonSerializerOptions.GetTypeInfo(typeof(JsonElement)))), + new TextContent("The new state is:") + ]); + + var firstRunMessages = messages.Append(stateUpdateMessage); + + // Collect all updates from first run + var allUpdates = new List(); + await foreach (var update in this.InnerAgent.RunStreamingAsync(firstRunMessages, thread, firstRunOptions, cancellationToken).ConfigureAwait(false)) + { + allUpdates.Add(update); + + // Yield all non-text updates (tool calls, etc.) + bool hasNonTextContent = update.Contents.Any(c => c is not TextContent); + if (hasNonTextContent) + { + yield return update; + } + } + + var response = allUpdates.ToAgentRunResponse(); + + // Try to deserialize the structured state response + if (response.TryDeserialize(this._jsonSerializerOptions, out JsonElement stateSnapshot)) + { + // Serialize and emit as STATE_SNAPSHOT via DataContent + byte[] stateBytes = JsonSerializer.SerializeToUtf8Bytes( + stateSnapshot, + this._jsonSerializerOptions.GetTypeInfo(typeof(JsonElement))); + yield return new AgentRunResponseUpdate + { + Contents = [new DataContent(stateBytes, "application/json")] + }; + } + else + { + yield break; + } + + // Second run: Generate user-friendly summary + var secondRunMessages = messages.Concat(response.Messages).Append( + new ChatMessage( + ChatRole.System, + [new TextContent("Please provide a concise summary of the state changes in at most two sentences.")])); + + await foreach (var update in this.InnerAgent.RunStreamingAsync(secondRunMessages, thread, options, cancellationToken).ConfigureAwait(false)) + { + yield return update; + } + } +} diff --git a/dotnet/samples/GettingStarted/AGUI/Step05_StateManagement/Server/appsettings.Development.json b/dotnet/samples/GettingStarted/AGUI/Step05_StateManagement/Server/appsettings.Development.json new file mode 100644 index 0000000000..0c208ae918 --- /dev/null +++ b/dotnet/samples/GettingStarted/AGUI/Step05_StateManagement/Server/appsettings.Development.json @@ -0,0 +1,8 @@ +{ + "Logging": { + "LogLevel": { + "Default": "Information", + "Microsoft.AspNetCore": "Warning" + } + } +} diff --git a/dotnet/samples/GettingStarted/AGUI/Step05_StateManagement/Server/appsettings.json b/dotnet/samples/GettingStarted/AGUI/Step05_StateManagement/Server/appsettings.json new file mode 100644 index 0000000000..10f68b8c8b --- /dev/null +++ b/dotnet/samples/GettingStarted/AGUI/Step05_StateManagement/Server/appsettings.json @@ -0,0 +1,9 @@ +{ + "Logging": { + "LogLevel": { + "Default": "Information", + "Microsoft.AspNetCore": "Warning" + } + }, + "AllowedHosts": "*" +} diff --git a/dotnet/samples/GettingStarted/AgentOpenTelemetry/AgentOpenTelemetry.csproj b/dotnet/samples/GettingStarted/AgentOpenTelemetry/AgentOpenTelemetry.csproj index f9b7b3da2a..e194fec9c2 100644 --- a/dotnet/samples/GettingStarted/AgentOpenTelemetry/AgentOpenTelemetry.csproj +++ b/dotnet/samples/GettingStarted/AgentOpenTelemetry/AgentOpenTelemetry.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable @@ -22,7 +22,6 @@ - diff --git a/dotnet/samples/GettingStarted/AgentOpenTelemetry/README.md b/dotnet/samples/GettingStarted/AgentOpenTelemetry/README.md index 3542bf5b30..229d37dca6 100644 --- a/dotnet/samples/GettingStarted/AgentOpenTelemetry/README.md +++ b/dotnet/samples/GettingStarted/AgentOpenTelemetry/README.md @@ -22,7 +22,7 @@ graph TD ## Prerequisites -- .NET 8.0 SDK or later +- .NET 10 SDK or later - Azure OpenAI service endpoint and deployment configured - Azure CLI installed and authenticated (for Azure credential authentication) - Docker installed (for running Aspire Dashboard) @@ -71,7 +71,7 @@ If you prefer to run the components manually: #### Step 1: Start the Aspire Dashboard via Docker ```powershell -docker run -d --name aspire-dashboard -p 4318:18888 -p 4317:18889 -e DOTNET_DASHBOARD_UNSECURED_ALLOW_ANONYMOUS=true mcr.microsoft.com/dotnet/aspire-dashboard:9.0 +docker run -d --name aspire-dashboard -p 4318:18888 -p 4317:18889 -e DOTNET_DASHBOARD_UNSECURED_ALLOW_ANONYMOUS=true mcr.microsoft.com/dotnet/aspire-dashboard:latest ``` #### Step 2: Access the Dashboard @@ -142,11 +142,11 @@ You: Besides the Aspire Dashboard and the Application Insights native UI, you can also use Grafana to visualize the telemetry data in Application Insights. There are two tailored dashboards for you to get started quickly: ### Agent Overview dashboard -Grafana Dashboard Gallery link: +Open dashboard in Azure portal: ![Agent Overview dashboard](https://github.com/Azure/azure-managed-grafana/raw/main/samples/assets/grafana-af-agent.gif) ### Workflow Overview dashboard -Grafana Dashboard Gallery link: +Open dashboard in Azure portal: ![Workflow Overview dashboard](https://github.com/Azure/azure-managed-grafana/raw/main/samples/assets/grafana-af-workflow.gif) ## Key Features Demonstrated @@ -207,7 +207,7 @@ If you encounter port binding errors, try: - Ensure the Azure OpenAI deployment name matches your actual deployment ### Build Issues -- Ensure you're using .NET 9.0 SDK +- Ensure you're using .NET 10.0 SDK - Run `dotnet restore` if you encounter package restore issues - Check that all project references are correctly resolved diff --git a/dotnet/samples/GettingStarted/AgentOpenTelemetry/start-demo.ps1 b/dotnet/samples/GettingStarted/AgentOpenTelemetry/start-demo.ps1 index 8445d1e7e3..7af1c9d8ae 100644 --- a/dotnet/samples/GettingStarted/AgentOpenTelemetry/start-demo.ps1 +++ b/dotnet/samples/GettingStarted/AgentOpenTelemetry/start-demo.ps1 @@ -65,7 +65,7 @@ $dockerResult = docker run -d ` -p 4317:18889 ` -e DOTNET_DASHBOARD_UNSECURED_ALLOW_ANONYMOUS=true ` --restart unless-stopped ` - mcr.microsoft.com/dotnet/aspire-dashboard:9.0 + mcr.microsoft.com/dotnet/aspire-dashboard:latest if ($LASTEXITCODE -ne 0) { Write-Host "Failed to start Aspire Dashboard container" -ForegroundColor Red diff --git a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_A2A/Agent_With_A2A.csproj b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_A2A/Agent_With_A2A.csproj index e01a9f7458..7236ee5044 100644 --- a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_A2A/Agent_With_A2A.csproj +++ b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_A2A/Agent_With_A2A.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable @@ -10,8 +10,6 @@ - - diff --git a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_A2A/README.md b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_A2A/README.md index ce7a9174b0..536514306e 100644 --- a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_A2A/README.md +++ b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_A2A/README.md @@ -2,7 +2,7 @@ Before you begin, ensure you have the following prerequisites: -- .NET 8.0 SDK or later +- .NET 10 SDK or later - Access to the A2A agent host service **Note**: These samples need to be run against a valid A2A server. If no A2A server is available, they can be run against the echo-agent that can be spun up locally by following the guidelines at: https://github.com/a2aproject/a2a-dotnet/blob/main/samples/AgentServer/README.md diff --git a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_Anthropic/Agent_With_Anthropic.csproj b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_Anthropic/Agent_With_Anthropic.csproj new file mode 100644 index 0000000000..eb29d1d310 --- /dev/null +++ b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_Anthropic/Agent_With_Anthropic.csproj @@ -0,0 +1,21 @@ + + + + Exe + net10.0 + + enable + enable + $(NoWarn);IDE0059 + + + + + + + + + + + + diff --git a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_Anthropic/Program.cs b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_Anthropic/Program.cs new file mode 100644 index 0000000000..df070c335b --- /dev/null +++ b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_Anthropic/Program.cs @@ -0,0 +1,74 @@ +// Copyright (c) Microsoft. All rights reserved. + +// This sample shows how to create and use an AI agent with Anthropic as the backend. + +using System.Net.Http.Headers; +using Anthropic; +using Anthropic.Foundry; +using Azure.Core; +using Azure.Identity; +using Microsoft.Agents.AI; +using Sample; + +var deploymentName = Environment.GetEnvironmentVariable("ANTHROPIC_DEPLOYMENT_NAME") ?? "claude-haiku-4-5"; + +// The resource is the subdomain name / first name coming before '.services.ai.azure.com' in the endpoint Uri +// ie: https://(resource name).services.ai.azure.com/anthropic/v1/chat/completions +string? resource = Environment.GetEnvironmentVariable("ANTHROPIC_RESOURCE"); +string? apiKey = Environment.GetEnvironmentVariable("ANTHROPIC_API_KEY"); + +const string JokerInstructions = "You are good at telling jokes."; +const string JokerName = "JokerAgent"; + +AnthropicClient? client = (resource is null) + ? new AnthropicClient() { APIKey = apiKey ?? throw new InvalidOperationException("ANTHROPIC_API_KEY is required when no ANTHROPIC_RESOURCE is provided") } // If no resource is provided, use Anthropic public API + : (apiKey is not null) + ? new AnthropicFoundryClient(new AnthropicFoundryApiKeyCredentials(apiKey, resource)) // If an apiKey is provided, use Foundry with ApiKey authentication + : new AnthropicFoundryClient(new AnthropicAzureTokenCredential(new AzureCliCredential(), resource)); // Otherwise, use Foundry with Azure Client authentication + +AIAgent agent = client.CreateAIAgent(model: deploymentName, instructions: JokerInstructions, name: JokerName); + +// Invoke the agent and output the text result. +Console.WriteLine(await agent.RunAsync("Tell me a joke about a pirate.")); + +namespace Sample +{ + /// + /// Provides methods for invoking the Azure hosted Anthropic models using types. + /// + public sealed class AnthropicAzureTokenCredential : IAnthropicFoundryCredentials + { + private readonly TokenCredential _tokenCredential; + private readonly Lock _lock = new(); + private AccessToken? _cachedAccessToken; + + /// + public string ResourceName { get; } + + /// + /// Creates a new instance of the . + /// + /// The credential provider. Use any specialization of to get your access token in supported environments. + /// The service resource subdomain name to use in the anthropic azure endpoint + internal AnthropicAzureTokenCredential(TokenCredential tokenCredential, string resourceName) + { + this.ResourceName = resourceName ?? throw new ArgumentNullException(nameof(resourceName)); + this._tokenCredential = tokenCredential ?? throw new ArgumentNullException(nameof(tokenCredential)); + } + + /// + public void Apply(HttpRequestMessage requestMessage) + { + lock (this._lock) + { + // Add a 5-minute buffer to avoid using tokens that are about to expire + if (this._cachedAccessToken is null || this._cachedAccessToken.Value.ExpiresOn <= DateTimeOffset.Now.AddMinutes(5)) + { + this._cachedAccessToken = this._tokenCredential.GetToken(new TokenRequestContext(scopes: ["https://ai.azure.com/.default"]), CancellationToken.None); + } + } + + requestMessage.Headers.Authorization = new AuthenticationHeaderValue("bearer", this._cachedAccessToken.Value.Token); + } + } +} diff --git a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_Anthropic/README.md b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_Anthropic/README.md new file mode 100644 index 0000000000..afcf391572 --- /dev/null +++ b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_Anthropic/README.md @@ -0,0 +1,53 @@ +# Creating an AIAgent with Anthropic + +This sample demonstrates how to create an AIAgent using Anthropic Claude models as the underlying inference service. + +The sample supports three deployment scenarios: + +1. **Anthropic Public API** - Direct connection to Anthropic's public API +2. **Azure Foundry with API Key** - Anthropic models deployed through Azure Foundry using API key authentication +3. **Azure Foundry with Azure CLI** - Anthropic models deployed through Azure Foundry using Azure CLI credentials + +## Prerequisites + +Before you begin, ensure you have the following prerequisites: + +- .NET 8.0 SDK or later + +### For Anthropic Public API + +- Anthropic API key + +Set the following environment variables: + +```powershell +$env:ANTHROPIC_API_KEY="your-anthropic-api-key" # Replace with your Anthropic API key +$env:ANTHROPIC_DEPLOYMENT_NAME="claude-haiku-4-5" # Optional, defaults to claude-haiku-4-5 +``` + +### For Azure Foundry with API Key + +- Azure Foundry service endpoint and deployment configured +- Anthropic API key + +Set the following environment variables: + +```powershell +$env:ANTHROPIC_RESOURCE="your-foundry-resource-name" # Replace with your Azure Foundry resource name (subdomain before .services.ai.azure.com) +$env:ANTHROPIC_API_KEY="your-anthropic-api-key" # Replace with your Anthropic API key +$env:ANTHROPIC_DEPLOYMENT_NAME="claude-haiku-4-5" # Optional, defaults to claude-haiku-4-5 +``` + +### For Azure Foundry with Azure CLI + +- Azure Foundry service endpoint and deployment configured +- Azure CLI installed and authenticated (for Azure credential authentication) + +Set the following environment variables: + +```powershell +$env:ANTHROPIC_RESOURCE="your-foundry-resource-name" # Replace with your Azure Foundry resource name (subdomain before .services.ai.azure.com) +$env:ANTHROPIC_DEPLOYMENT_NAME="claude-haiku-4-5" # Optional, defaults to claude-haiku-4-5 +``` + +**Note**: When using Azure Foundry with Azure CLI, make sure you're logged in with `az login` and have access to the Azure Foundry resource. For more information, see the [Azure CLI documentation](https://learn.microsoft.com/cli/azure/authenticate-azure-cli-interactively). diff --git a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureFoundryAgent/Agent_With_AzureFoundryAgent.csproj b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureAIAgentsPersistent/Agent_With_AzureAIAgentsPersistent.csproj similarity index 91% rename from dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureFoundryAgent/Agent_With_AzureFoundryAgent.csproj rename to dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureAIAgentsPersistent/Agent_With_AzureAIAgentsPersistent.csproj index 11c7beb3bf..d40e93232b 100644 --- a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureFoundryAgent/Agent_With_AzureFoundryAgent.csproj +++ b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureAIAgentsPersistent/Agent_With_AzureAIAgentsPersistent.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable diff --git a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureFoundryAgent/Program.cs b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureAIAgentsPersistent/Program.cs similarity index 100% rename from dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureFoundryAgent/Program.cs rename to dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureAIAgentsPersistent/Program.cs diff --git a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureAIAgentsPersistent/README.md b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureAIAgentsPersistent/README.md new file mode 100644 index 0000000000..d6b5497601 --- /dev/null +++ b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureAIAgentsPersistent/README.md @@ -0,0 +1,26 @@ +# Classic Foundry Agents + +This sample demonstrates how to create an agent using the classic Foundry Agents experience. + +# Classic vs New Foundry Agents + +Below is a comparison between the classic and new Foundry Agents approaches: + +[Migration Guide](https://learn.microsoft.com/en-us/azure/ai-foundry/agents/how-to/migrate?view=foundry) + +# Prerequisites + +Before you begin, ensure you have the following prerequisites: + +- .NET 10 SDK or later +- Azure Foundry service endpoint and deployment configured +- Azure CLI installed and authenticated (for Azure credential authentication) + +**Note**: This demo uses Azure CLI credentials for authentication. Make sure you're logged in with `az login` and have access to the Azure Foundry resource. For more information, see the [Azure CLI documentation](https://learn.microsoft.com/cli/azure/authenticate-azure-cli-interactively). + +Set the following environment variables: + +```powershell +$env:AZURE_FOUNDRY_PROJECT_ENDPOINT="https://your-foundry-service.services.ai.azure.com/api/projects/your-foundry-project" # Replace with your Azure Foundry resource endpoint +$env:AZURE_FOUNDRY_PROJECT_DEPLOYMENT_NAME="gpt-4o-mini" # Optional, defaults to gpt-4o-mini +``` diff --git a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureAIProject/Agent_With_AzureAIProject.csproj b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureAIProject/Agent_With_AzureAIProject.csproj new file mode 100644 index 0000000000..a8deaa57b5 --- /dev/null +++ b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureAIProject/Agent_With_AzureAIProject.csproj @@ -0,0 +1,21 @@ + + + + Exe + net10.0 + + enable + enable + $(NoWarn);IDE0059 + + + + + + + + + + + + diff --git a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureAIProject/Program.cs b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureAIProject/Program.cs new file mode 100644 index 0000000000..2c2b9d1969 --- /dev/null +++ b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureAIProject/Program.cs @@ -0,0 +1,50 @@ +// Copyright (c) Microsoft. All rights reserved. + +// This sample shows how to create and use a AI agents with Azure Foundry Agents as the backend. + +using Azure.AI.Projects; +using Azure.AI.Projects.OpenAI; +using Azure.Identity; +using Microsoft.Agents.AI; + +var endpoint = Environment.GetEnvironmentVariable("AZURE_FOUNDRY_PROJECT_ENDPOINT") ?? throw new InvalidOperationException("AZURE_FOUNDRY_PROJECT_ENDPOINT is not set."); +var deploymentName = Environment.GetEnvironmentVariable("AZURE_FOUNDRY_PROJECT_DEPLOYMENT_NAME") ?? "gpt-4o-mini"; + +const string JokerName = "JokerAgent"; + +// Get a client to create/retrieve/delete server side agents with Azure Foundry Agents. +var aiProjectClient = new AIProjectClient(new Uri(endpoint), new AzureCliCredential()); + +// Define the agent you want to create. (Prompt Agent in this case) +var agentVersionCreationOptions = new AgentVersionCreationOptions(new PromptAgentDefinition(model: deploymentName) { Instructions = "You are good at telling jokes." }); +// Azure.AI.Agents SDK creates and manages agent by name and versions. +// You can create a server side agent version with the Azure.AI.Agents SDK client below. +var createdAgentVersion = aiProjectClient.Agents.CreateAgentVersion(agentName: JokerName, options: agentVersionCreationOptions); + +// Note: +// agentVersion.Id = ":", +// agentVersion.Version = , +// agentVersion.Name = + +// You can retrieve an AIAgent for an already created server side agent version. +AIAgent existingJokerAgent = aiProjectClient.GetAIAgent(createdAgentVersion); + +// You can also create another AIAgent version by providing the same name with a different definition. +AIAgent newJokerAgent = aiProjectClient.CreateAIAgent(name: JokerName, model: deploymentName, instructions: "You are extremely hilarious at telling jokes."); + +// You can also get the AIAgent latest version just providing its name. +AIAgent jokerAgentLatest = aiProjectClient.GetAIAgent(name: JokerName); +var latestAgentVersion = jokerAgentLatest.GetService()!; + +// The AIAgent version can be accessed via the GetService method. +Console.WriteLine($"Latest agent version id: {latestAgentVersion.Id}"); + +// Once you have the AIAgent, you can invoke it like any other AIAgent. +AgentThread thread = jokerAgentLatest.GetNewThread(); +Console.WriteLine(await jokerAgentLatest.RunAsync("Tell me a joke about a pirate.", thread)); + +// This will use the same thread to continue the conversation. +Console.WriteLine(await jokerAgentLatest.RunAsync("Now tell me a joke about a cat and a dog using last joke as the anchor.", thread)); + +// Cleanup by agent name removes both agent versions created. +aiProjectClient.Agents.DeleteAgent(existingJokerAgent.Name); diff --git a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureFoundryAgent/README.md b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureAIProject/README.md similarity index 70% rename from dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureFoundryAgent/README.md rename to dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureAIProject/README.md index df0854ba2f..7e4a28f6a1 100644 --- a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureFoundryAgent/README.md +++ b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureAIProject/README.md @@ -1,8 +1,18 @@ +# New Foundry Agents + +This sample demonstrates how to create an agent using the new Foundry Agents experience. + +# Classic vs New Foundry Agents + +Below is a comparison between the classic and new Foundry Agents approaches: + +[Migration Guide](https://learn.microsoft.com/en-us/azure/ai-foundry/agents/how-to/migrate?view=foundry) + # Prerequisites Before you begin, ensure you have the following prerequisites: -- .NET 8.0 SDK or later +- .NET 10 SDK or later - Azure Foundry service endpoint and deployment configured - Azure CLI installed and authenticated (for Azure credential authentication) diff --git a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureFoundryModel/Agent_With_AzureFoundryModel.csproj b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureFoundryModel/Agent_With_AzureFoundryModel.csproj index cd545ddb48..0c4701fafd 100644 --- a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureFoundryModel/Agent_With_AzureFoundryModel.csproj +++ b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureFoundryModel/Agent_With_AzureFoundryModel.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable diff --git a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureFoundryModel/Program.cs b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureFoundryModel/Program.cs index 264a9e45e8..5ed66d3e4b 100644 --- a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureFoundryModel/Program.cs +++ b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureFoundryModel/Program.cs @@ -9,9 +9,10 @@ using Azure.Identity; using Microsoft.Agents.AI; using OpenAI; +using OpenAI.Chat; var endpoint = Environment.GetEnvironmentVariable("AZURE_FOUNDRY_OPENAI_ENDPOINT") ?? throw new InvalidOperationException("AZURE_FOUNDRY_OPENAI_ENDPOINT is not set."); -var apiKey = Environment.GetEnvironmentVariable("AZURE_FOUNDRY_OPENAI_APIKEY"); +var apiKey = Environment.GetEnvironmentVariable("AZURE_FOUNDRY_OPENAI_API_KEY"); var model = Environment.GetEnvironmentVariable("AZURE_FOUNDRY_MODEL_DEPLOYMENT") ?? "Phi-4-mini-instruct"; // Since we are using the OpenAI Client SDK, we need to override the default endpoint to point to Azure Foundry. diff --git a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureFoundryModel/README.md b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureFoundryModel/README.md index 9147bda1da..b5e65ea209 100644 --- a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureFoundryModel/README.md +++ b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureFoundryModel/README.md @@ -10,7 +10,7 @@ You could use models from Microsoft, OpenAI, DeepSeek, Hugging Face, Meta, xAI o Before you begin, ensure you have the following prerequisites: -- .NET 8.0 SDK or later +- .NET 10 SDK or later - Azure AI Foundry resource - A model deployment in your Azure AI Foundry resource. This example defaults to using the `Phi-4-mini-instruct` model, so if you want to use a different model, ensure that you set your `AZURE_FOUNDRY_MODEL_DEPLOYMENT` environment @@ -27,7 +27,7 @@ Set the following environment variables: $env:AZURE_FOUNDRY_OPENAI_ENDPOINT="https://ai-foundry-.services.ai.azure.com/openai/v1/" # Optional, defaults to using Azure CLI for authentication if not provided -$env:AZURE_FOUNDRY_OPENAI_APIKEY="************" +$env:AZURE_FOUNDRY_OPENAI_API_KEY="************" # Optional, defaults to Phi-4-mini-instruct $env:AZURE_FOUNDRY_MODEL_DEPLOYMENT="Phi-4-mini-instruct" diff --git a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureOpenAIChatCompletion/Agent_With_AzureOpenAIChatCompletion.csproj b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureOpenAIChatCompletion/Agent_With_AzureOpenAIChatCompletion.csproj index 0eacdab258..41aafe3437 100644 --- a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureOpenAIChatCompletion/Agent_With_AzureOpenAIChatCompletion.csproj +++ b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureOpenAIChatCompletion/Agent_With_AzureOpenAIChatCompletion.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable diff --git a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureOpenAIChatCompletion/Program.cs b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureOpenAIChatCompletion/Program.cs index bd31350258..cf717550d2 100644 --- a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureOpenAIChatCompletion/Program.cs +++ b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureOpenAIChatCompletion/Program.cs @@ -5,7 +5,7 @@ using Azure.AI.OpenAI; using Azure.Identity; using Microsoft.Agents.AI; -using OpenAI; +using OpenAI.Chat; var endpoint = Environment.GetEnvironmentVariable("AZURE_OPENAI_ENDPOINT") ?? throw new InvalidOperationException("AZURE_OPENAI_ENDPOINT is not set."); var deploymentName = Environment.GetEnvironmentVariable("AZURE_OPENAI_DEPLOYMENT_NAME") ?? "gpt-4o-mini"; diff --git a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureOpenAIChatCompletion/README.md b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureOpenAIChatCompletion/README.md index 1278eb59e5..4cacf30131 100644 --- a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureOpenAIChatCompletion/README.md +++ b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureOpenAIChatCompletion/README.md @@ -2,7 +2,7 @@ Before you begin, ensure you have the following prerequisites: -- .NET 8.0 SDK or later +- .NET 10 SDK or later - Azure OpenAI service endpoint and deployment configured - Azure CLI installed and authenticated (for Azure credential authentication) diff --git a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureOpenAIResponses/Agent_With_AzureOpenAIResponses.csproj b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureOpenAIResponses/Agent_With_AzureOpenAIResponses.csproj index 0eacdab258..41aafe3437 100644 --- a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureOpenAIResponses/Agent_With_AzureOpenAIResponses.csproj +++ b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureOpenAIResponses/Agent_With_AzureOpenAIResponses.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable diff --git a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureOpenAIResponses/Program.cs b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureOpenAIResponses/Program.cs index 6d162ebfd6..83d5619382 100644 --- a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureOpenAIResponses/Program.cs +++ b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureOpenAIResponses/Program.cs @@ -5,7 +5,7 @@ using Azure.AI.OpenAI; using Azure.Identity; using Microsoft.Agents.AI; -using OpenAI; +using OpenAI.Responses; var endpoint = Environment.GetEnvironmentVariable("AZURE_OPENAI_ENDPOINT") ?? throw new InvalidOperationException("AZURE_OPENAI_ENDPOINT is not set."); var deploymentName = Environment.GetEnvironmentVariable("AZURE_OPENAI_DEPLOYMENT_NAME") ?? "gpt-4o-mini"; diff --git a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureOpenAIResponses/README.md b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureOpenAIResponses/README.md index 1278eb59e5..4cacf30131 100644 --- a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureOpenAIResponses/README.md +++ b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_AzureOpenAIResponses/README.md @@ -2,7 +2,7 @@ Before you begin, ensure you have the following prerequisites: -- .NET 8.0 SDK or later +- .NET 10 SDK or later - Azure OpenAI service endpoint and deployment configured - Azure CLI installed and authenticated (for Azure credential authentication) diff --git a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_CustomImplementation/Agent_With_CustomImplementation.csproj b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_CustomImplementation/Agent_With_CustomImplementation.csproj index aa1c382aef..945912bfd4 100644 --- a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_CustomImplementation/Agent_With_CustomImplementation.csproj +++ b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_CustomImplementation/Agent_With_CustomImplementation.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable diff --git a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_CustomImplementation/Program.cs b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_CustomImplementation/Program.cs index fd00618f5f..8f1039251d 100644 --- a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_CustomImplementation/Program.cs +++ b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_CustomImplementation/Program.cs @@ -39,11 +39,16 @@ public override async Task RunAsync(IEnumerable m // Create a thread if the user didn't supply one. thread ??= this.GetNewThread(); + if (thread is not CustomAgentThread typedThread) + { + throw new ArgumentException($"The provided thread is not of type {nameof(CustomAgentThread)}.", nameof(thread)); + } + // Clone the input messages and turn them into response messages with upper case text. List responseMessages = CloneAndToUpperCase(messages, this.DisplayName).ToList(); // Notify the thread of the input and output messages. - await NotifyThreadOfNewMessagesAsync(thread, messages.Concat(responseMessages), cancellationToken); + await typedThread.MessageStore.AddMessagesAsync(messages.Concat(responseMessages), cancellationToken); return new AgentRunResponse { @@ -58,11 +63,16 @@ public override async IAsyncEnumerable RunStreamingAsync // Create a thread if the user didn't supply one. thread ??= this.GetNewThread(); + if (thread is not CustomAgentThread typedThread) + { + throw new ArgumentException($"The provided thread is not of type {nameof(CustomAgentThread)}.", nameof(thread)); + } + // Clone the input messages and turn them into response messages with upper case text. List responseMessages = CloneAndToUpperCase(messages, this.DisplayName).ToList(); // Notify the thread of the input and output messages. - await NotifyThreadOfNewMessagesAsync(thread, messages.Concat(responseMessages), cancellationToken); + await typedThread.MessageStore.AddMessagesAsync(messages.Concat(responseMessages), cancellationToken); foreach (var message in responseMessages) { diff --git a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_GoogleGemini/Agent_With_GoogleGemini.csproj b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_GoogleGemini/Agent_With_GoogleGemini.csproj new file mode 100644 index 0000000000..d01f015a4b --- /dev/null +++ b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_GoogleGemini/Agent_With_GoogleGemini.csproj @@ -0,0 +1,25 @@ + + + + Exe + net8.0;net9.0;net10.0 + + enable + enable + $(NoWarn);IDE0059;NU1510 + + + + + + + + + + + + + + + + diff --git a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_GoogleGemini/GeminiChatClient.cs b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_GoogleGemini/GeminiChatClient.cs new file mode 100644 index 0000000000..28f6f26013 --- /dev/null +++ b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_GoogleGemini/GeminiChatClient.cs @@ -0,0 +1,558 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Runtime.CompilerServices; +using Google.Apis.Util; +using Google.GenAI; +using Google.GenAI.Types; + +namespace Microsoft.Extensions.AI; + +/// Provides an implementation based on . +internal sealed class GoogleGenAIChatClient : IChatClient +{ + /// The wrapped instance (optional). + private readonly Client? _client; + + /// The wrapped instance. + private readonly Models _models; + + /// The default model that should be used when no override is specified. + private readonly string? _defaultModelId; + + /// Lazily-initialized metadata describing the implementation. + private ChatClientMetadata? _metadata; + + /// Initializes a new instance. + public GoogleGenAIChatClient(Client client, string? defaultModelId) + { + this._client = client; + this._models = client.Models; + this._defaultModelId = defaultModelId; + } + + /// Initializes a new instance. + public GoogleGenAIChatClient(Models client, string? defaultModelId) + { + this._models = client; + this._defaultModelId = defaultModelId; + } + + /// + public async Task GetResponseAsync(IEnumerable messages, ChatOptions? options = null, CancellationToken cancellationToken = default) + { + Utilities.ThrowIfNull(messages, nameof(messages)); + + // Create the request. + (string? modelId, List contents, GenerateContentConfig config) = this.CreateRequest(messages, options); + + // Send it. + GenerateContentResponse generateResult = await this._models.GenerateContentAsync(modelId!, contents, config).ConfigureAwait(false); + + // Create the response. + ChatResponse chatResponse = new(new ChatMessage(ChatRole.Assistant, [])) + { + CreatedAt = generateResult.CreateTime is { } dt ? new DateTimeOffset(dt) : null, + ModelId = !string.IsNullOrWhiteSpace(generateResult.ModelVersion) ? generateResult.ModelVersion : modelId, + RawRepresentation = generateResult, + ResponseId = generateResult.ResponseId, + }; + + // Populate the response messages. + chatResponse.FinishReason = PopulateResponseContents(generateResult, chatResponse.Messages[0].Contents); + + // Populate usage information if there is any. + if (generateResult.UsageMetadata is { } usageMetadata) + { + chatResponse.Usage = ExtractUsageDetails(usageMetadata); + } + + // Return the response. + return chatResponse; + } + + /// + public async IAsyncEnumerable GetStreamingResponseAsync(IEnumerable messages, ChatOptions? options = null, [EnumeratorCancellation] CancellationToken cancellationToken = default) + { + Utilities.ThrowIfNull(messages, nameof(messages)); + + // Create the request. + (string? modelId, List contents, GenerateContentConfig config) = this.CreateRequest(messages, options); + + // Send it, and process the results. + await foreach (GenerateContentResponse generateResult in this._models.GenerateContentStreamAsync(modelId!, contents, config).WithCancellation(cancellationToken).ConfigureAwait(false)) + { + // Create a response update for each result in the stream. + ChatResponseUpdate responseUpdate = new(ChatRole.Assistant, []) + { + CreatedAt = generateResult.CreateTime is { } dt ? new DateTimeOffset(dt) : null, + ModelId = !string.IsNullOrWhiteSpace(generateResult.ModelVersion) ? generateResult.ModelVersion : modelId, + RawRepresentation = generateResult, + ResponseId = generateResult.ResponseId, + }; + + // Populate the response update contents. + responseUpdate.FinishReason = PopulateResponseContents(generateResult, responseUpdate.Contents); + + // Populate usage information if there is any. + if (generateResult.UsageMetadata is { } usageMetadata) + { + responseUpdate.Contents.Add(new UsageContent(ExtractUsageDetails(usageMetadata))); + } + + // Yield the update. + yield return responseUpdate; + } + } + + /// + public object? GetService(System.Type serviceType, object? serviceKey = null) + { + Utilities.ThrowIfNull(serviceType, nameof(serviceType)); + + if (serviceKey is null) + { + // If there's a request for metadata, lazily-initialize it and return it. We don't need to worry about race conditions, + // as there's no requirement that the same instance be returned each time, and creation is idempotent. + if (serviceType == typeof(ChatClientMetadata)) + { + return this._metadata ??= new("gcp.gen_ai", new("https://generativelanguage.googleapis.com/"), defaultModelId: this._defaultModelId); + } + + // Allow a consumer to "break glass" and access the underlying client if they need it. + if (serviceType.IsInstanceOfType(this._models)) + { + return this._models; + } + + if (this._client is not null && serviceType.IsInstanceOfType(this._client)) + { + return this._client; + } + + if (serviceType.IsInstanceOfType(this)) + { + return this; + } + } + + return null; + } + + /// + void IDisposable.Dispose() { /* nop */ } + + /// Creates the message parameters for from and . + private (string? ModelId, List Contents, GenerateContentConfig Config) CreateRequest(IEnumerable messages, ChatOptions? options) + { + // Create the GenerateContentConfig object. If the options contains a RawRepresentationFactory, try to use it to + // create the request instance, allowing the caller to populate it with GenAI-specific options. Otherwise, create + // a new instance directly. + string? model = this._defaultModelId; + List contents = []; + GenerateContentConfig config = options?.RawRepresentationFactory?.Invoke(this) as GenerateContentConfig ?? new(); + + if (options is not null) + { + if (options.FrequencyPenalty is { } frequencyPenalty) + { + config.FrequencyPenalty ??= frequencyPenalty; + } + + if (options.Instructions is { } instructions) + { + ((config.SystemInstruction ??= new()).Parts ??= []).Add(new() { Text = instructions }); + } + + if (options.MaxOutputTokens is { } maxOutputTokens) + { + config.MaxOutputTokens ??= maxOutputTokens; + } + + if (!string.IsNullOrWhiteSpace(options.ModelId)) + { + model = options.ModelId; + } + + if (options.PresencePenalty is { } presencePenalty) + { + config.PresencePenalty ??= presencePenalty; + } + + if (options.Seed is { } seed) + { + config.Seed ??= (int)seed; + } + + if (options.StopSequences is { } stopSequences) + { + (config.StopSequences ??= []).AddRange(stopSequences); + } + + if (options.Temperature is { } temperature) + { + config.Temperature ??= temperature; + } + + if (options.TopP is { } topP) + { + config.TopP ??= topP; + } + + if (options.TopK is { } topK) + { + config.TopK ??= topK; + } + + // Populate tools. Each kind of tool is added on its own, except for function declarations, + // which are grouped into a single FunctionDeclaration. + List? functionDeclarations = null; + if (options.Tools is { } tools) + { + foreach (var tool in tools) + { + switch (tool) + { + case AIFunctionDeclaration af: + functionDeclarations ??= []; + functionDeclarations.Add(new() + { + Name = af.Name, + Description = af.Description ?? "", + ParametersJsonSchema = af.JsonSchema, + }); + break; + + case HostedCodeInterpreterTool: + (config.Tools ??= []).Add(new() { CodeExecution = new() }); + break; + + case HostedFileSearchTool: + (config.Tools ??= []).Add(new() { Retrieval = new() }); + break; + + case HostedWebSearchTool: + (config.Tools ??= []).Add(new() { GoogleSearch = new() }); + break; + } + } + } + + if (functionDeclarations is { Count: > 0 }) + { + Tool functionTools = new(); + (functionTools.FunctionDeclarations ??= []).AddRange(functionDeclarations); + (config.Tools ??= []).Add(functionTools); + } + + // Transfer over the tool mode if there are any tools. + if (options.ToolMode is { } toolMode && config.Tools?.Count > 0) + { + switch (toolMode) + { + case NoneChatToolMode: + config.ToolConfig = new() { FunctionCallingConfig = new() { Mode = FunctionCallingConfigMode.NONE } }; + break; + + case AutoChatToolMode: + config.ToolConfig = new() { FunctionCallingConfig = new() { Mode = FunctionCallingConfigMode.AUTO } }; + break; + + case RequiredChatToolMode required: + config.ToolConfig = new() { FunctionCallingConfig = new() { Mode = FunctionCallingConfigMode.ANY } }; + if (required.RequiredFunctionName is not null) + { + ((config.ToolConfig.FunctionCallingConfig ??= new()).AllowedFunctionNames ??= []).Add(required.RequiredFunctionName); + } + break; + } + } + + // Set the response format if specified. + if (options.ResponseFormat is ChatResponseFormatJson responseFormat) + { + config.ResponseMimeType = "application/json"; + if (responseFormat.Schema is { } schema) + { + config.ResponseJsonSchema = schema; + } + } + } + + // Transfer messages to request, handling system messages specially + Dictionary? callIdToFunctionNames = null; + foreach (var message in messages) + { + if (message.Role == ChatRole.System) + { + string instruction = message.Text; + if (!string.IsNullOrWhiteSpace(instruction)) + { + ((config.SystemInstruction ??= new()).Parts ??= []).Add(new() { Text = instruction }); + } + + continue; + } + + Content content = new() { Role = message.Role == ChatRole.Assistant ? "model" : "user" }; + content.Parts ??= []; + AddPartsForAIContents(ref callIdToFunctionNames, message.Contents, content.Parts); + + contents.Add(content); + } + + // Make sure the request contains at least one content part (the request would always fail if empty). + if (!contents.SelectMany(c => c.Parts ?? Enumerable.Empty()).Any()) + { + contents.Add(new() { Role = "user", Parts = new() { { new() { Text = "" } } } }); + } + + return (model, contents, config); + } + + /// Creates s for and adds them to . + private static void AddPartsForAIContents(ref Dictionary? callIdToFunctionNames, IList contents, List parts) + { + for (int i = 0; i < contents.Count; i++) + { + var content = contents[i]; + + byte[]? thoughtSignature = null; + if (content is not TextReasoningContent { ProtectedData: not null } && + i + 1 < contents.Count && + contents[i + 1] is TextReasoningContent nextReasoning && + string.IsNullOrWhiteSpace(nextReasoning.Text) && + nextReasoning.ProtectedData is { } protectedData) + { + i++; + thoughtSignature = Convert.FromBase64String(protectedData); + } + + Part? part = null; + switch (content) + { + case TextContent textContent: + part = new() { Text = textContent.Text }; + break; + + case TextReasoningContent reasoningContent: + part = new() + { + Thought = true, + Text = !string.IsNullOrWhiteSpace(reasoningContent.Text) ? reasoningContent.Text : null, + ThoughtSignature = reasoningContent.ProtectedData is not null ? Convert.FromBase64String(reasoningContent.ProtectedData) : null, + }; + break; + + case DataContent dataContent: + part = new() + { + InlineData = new() + { + MimeType = dataContent.MediaType, + Data = dataContent.Data.ToArray(), + DisplayName = dataContent.Name, + } + }; + break; + + case UriContent uriContent: + part = new() + { + FileData = new() + { + FileUri = uriContent.Uri.AbsoluteUri, + MimeType = uriContent.MediaType, + } + }; + break; + + case FunctionCallContent functionCallContent: + (callIdToFunctionNames ??= [])[functionCallContent.CallId] = functionCallContent.Name; + callIdToFunctionNames[""] = functionCallContent.Name; // track last function name in case calls don't have IDs + + part = new() + { + FunctionCall = new() + { + Id = functionCallContent.CallId, + Name = functionCallContent.Name, + Args = functionCallContent.Arguments is null ? null : functionCallContent.Arguments as Dictionary ?? new(functionCallContent.Arguments!), + } + }; + break; + + case FunctionResultContent functionResultContent: + part = new() + { + FunctionResponse = new() + { + Id = functionResultContent.CallId, + Name = callIdToFunctionNames?.TryGetValue(functionResultContent.CallId, out string? functionName) is true || callIdToFunctionNames?.TryGetValue("", out functionName) is true ? + functionName : + null, + Response = functionResultContent.Result is null ? null : new() { ["result"] = functionResultContent.Result }, + } + }; + break; + } + + if (part is not null) + { + part.ThoughtSignature ??= thoughtSignature; + parts.Add(part); + } + } + } + + /// Creates s for and adds them to . + private static void AddAIContentsForParts(List parts, IList contents) + { + foreach (var part in parts) + { + AIContent? content = null; + + if (!string.IsNullOrEmpty(part.Text)) + { + content = part.Thought is true ? + new TextReasoningContent(part.Text) : + new TextContent(part.Text); + } + else if (part.InlineData is { } inlineData) + { + content = new DataContent(inlineData.Data, inlineData.MimeType ?? "application/octet-stream") + { + Name = inlineData.DisplayName, + }; + } + else if (part.FileData is { FileUri: not null } fileData) + { + content = new UriContent(new Uri(fileData.FileUri), fileData.MimeType ?? "application/octet-stream"); + } + else if (part.FunctionCall is { Name: not null } functionCall) + { + content = new FunctionCallContent(functionCall.Id ?? "", functionCall.Name, functionCall.Args!); + } + else if (part.FunctionResponse is { } functionResponse) + { + content = new FunctionResultContent( + functionResponse.Id ?? "", + functionResponse.Response?.TryGetValue("output", out var output) is true ? output : + functionResponse.Response?.TryGetValue("error", out var error) is true ? error : + null); + } + + if (content is not null) + { + content.RawRepresentation = part; + contents.Add(content); + + if (part.ThoughtSignature is { } thoughtSignature) + { + contents.Add(new TextReasoningContent(null) + { + ProtectedData = Convert.ToBase64String(thoughtSignature), + }); + } + } + } + } + + private static ChatFinishReason? PopulateResponseContents(GenerateContentResponse generateResult, IList responseContents) + { + ChatFinishReason? finishReason = null; + + // Populate the response messages. There should only be at most one candidate, but if there are more, ignore all but the first. + if (generateResult.Candidates is { Count: > 0 } && + generateResult.Candidates[0] is { Content: { } candidateContent } candidate) + { + // Grab the finish reason if one exists. + finishReason = ConvertFinishReason(candidate.FinishReason); + + // Add all of the response content parts as AIContents. + if (candidateContent.Parts is { } parts) + { + AddAIContentsForParts(parts, responseContents); + } + + // Add any citation metadata. + if (candidate.CitationMetadata is { Citations: { Count: > 0 } citations } && + responseContents.OfType().FirstOrDefault() is TextContent textContent) + { + foreach (var citation in citations) + { + textContent.Annotations = + [ + new CitationAnnotation() + { + Title = citation.Title, + Url = Uri.TryCreate(citation.Uri, UriKind.Absolute, out Uri? uri) ? uri : null, + AnnotatedRegions = + [ + new TextSpanAnnotatedRegion() + { + StartIndex = citation.StartIndex, + EndIndex = citation.EndIndex, + } + ], + } + ]; + } + } + } + + // Populate error information if there is any. + if (generateResult.PromptFeedback is { } promptFeedback) + { + responseContents.Add(new ErrorContent(promptFeedback.BlockReasonMessage)); + } + + return finishReason; + } + + /// Creates an M.E.AI from a Google . + private static ChatFinishReason? ConvertFinishReason(FinishReason? finishReason) + { + return finishReason switch + { + null => null, + + FinishReason.MAX_TOKENS => + ChatFinishReason.Length, + + FinishReason.MALFORMED_FUNCTION_CALL or + FinishReason.UNEXPECTED_TOOL_CALL => + ChatFinishReason.ToolCalls, + + FinishReason.FINISH_REASON_UNSPECIFIED or + FinishReason.STOP => + ChatFinishReason.Stop, + + _ => ChatFinishReason.ContentFilter, + }; + } + + /// Creates a populated from the supplied . + private static UsageDetails ExtractUsageDetails(GenerateContentResponseUsageMetadata usageMetadata) + { + UsageDetails details = new() + { + InputTokenCount = usageMetadata.PromptTokenCount, + OutputTokenCount = usageMetadata.CandidatesTokenCount, + TotalTokenCount = usageMetadata.TotalTokenCount, + }; + + AddIfPresent(nameof(usageMetadata.CachedContentTokenCount), usageMetadata.CachedContentTokenCount); + AddIfPresent(nameof(usageMetadata.ThoughtsTokenCount), usageMetadata.ThoughtsTokenCount); + AddIfPresent(nameof(usageMetadata.ToolUsePromptTokenCount), usageMetadata.ToolUsePromptTokenCount); + + return details; + + void AddIfPresent(string key, int? value) + { + if (value is int i) + { + (details.AdditionalCounts ??= [])[key] = i; + } + } + } +} diff --git a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_GoogleGemini/GoogleGenAIExtensions.cs b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_GoogleGemini/GoogleGenAIExtensions.cs new file mode 100644 index 0000000000..b1044fa373 --- /dev/null +++ b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_GoogleGemini/GoogleGenAIExtensions.cs @@ -0,0 +1,36 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Google.Apis.Util; +using Google.GenAI; + +namespace Microsoft.Extensions.AI; + +/// Provides implementations of Microsoft.Extensions.AI abstractions based on . +public static class GoogleGenAIExtensions +{ + /// + /// Creates an wrapper around the specified . + /// + /// The to wrap. + /// The default model ID to use for chat requests if not specified in . + /// An that wraps the specified client. + /// is . + public static IChatClient AsIChatClient(this Client client, string? defaultModelId = null) + { + Utilities.ThrowIfNull(client, nameof(client)); + return new GoogleGenAIChatClient(client, defaultModelId); + } + + /// + /// Creates an wrapper around the specified . + /// + /// The client to wrap. + /// The default model ID to use for chat requests if not specified in . + /// An that wraps the specified client. + /// is . + public static IChatClient AsIChatClient(this Models models, string? defaultModelId = null) + { + Utilities.ThrowIfNull(models, nameof(models)); + return new GoogleGenAIChatClient(models, defaultModelId); + } +} diff --git a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_GoogleGemini/Program.cs b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_GoogleGemini/Program.cs new file mode 100644 index 0000000000..89c86d5c56 --- /dev/null +++ b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_GoogleGemini/Program.cs @@ -0,0 +1,36 @@ +// Copyright (c) Microsoft. All rights reserved. + +// This sample shows how to create and use an AI agent with Google Gemini + +using Google.GenAI; +using Microsoft.Agents.AI; +using Microsoft.Extensions.AI; +using Mscc.GenerativeAI.Microsoft; + +const string JokerInstructions = "You are good at telling jokes."; +const string JokerName = "JokerAgent"; + +string apiKey = Environment.GetEnvironmentVariable("GOOGLE_GENAI_API_KEY") ?? throw new InvalidOperationException("Please set the GOOGLE_GENAI_API_KEY environment variable."); +string model = Environment.GetEnvironmentVariable("GOOGLE_GENAI_MODEL") ?? "gemini-2.5-flash"; + +// Using a Google GenAI IChatClient implementation +// Until the PR https://github.com/googleapis/dotnet-genai/pull/81 is not merged this option +// requires usage of also both GeminiChatClient.cs and GoogleGenAIExtensions.cs polyfills to work. + +ChatClientAgent agentGenAI = new( + new Client(vertexAI: false, apiKey: apiKey).AsIChatClient(model), + name: JokerName, + instructions: JokerInstructions); + +AgentRunResponse response = await agentGenAI.RunAsync("Tell me a joke about a pirate."); +Console.WriteLine($"Google GenAI client based agent response:\n{response}"); + +// Using a community driven Mscc.GenerativeAI.Microsoft package + +ChatClientAgent agentCommunity = new( + new GeminiChatClient(apiKey: apiKey, model: model), + name: JokerName, + instructions: JokerInstructions); + +response = await agentCommunity.RunAsync("Tell me a joke about a pirate."); +Console.WriteLine($"Community client based agent response:\n{response}"); diff --git a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_GoogleGemini/README.md b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_GoogleGemini/README.md new file mode 100644 index 0000000000..bc3a3592e6 --- /dev/null +++ b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_GoogleGemini/README.md @@ -0,0 +1,37 @@ +# Creating an AIAgent with Google Gemini + +This sample demonstrates how to create an AIAgent using Google Gemini models as the underlying inference service. + +The sample showcases two different `IChatClient` implementations: + +1. **Google GenAI** - Using the official [Google.GenAI](https://www.nuget.org/packages/Google.GenAI) package +2. **Mscc.GenerativeAI.Microsoft** - Using the community-driven [Mscc.GenerativeAI.Microsoft](https://www.nuget.org/packages/Mscc.GenerativeAI.Microsoft) package + +## Prerequisites + +Before you begin, ensure you have the following prerequisites: + +- .NET 10.0 SDK or later +- Google AI Studio API key (get one at [Google AI Studio](https://aistudio.google.com/apikey)) + +Set the following environment variables: + +```powershell +$env:GOOGLE_GENAI_API_KEY="your-google-api-key" # Replace with your Google AI Studio API key +$env:GOOGLE_GENAI_MODEL="gemini-2.5-fast" # Optional, defaults to gemini-2.5-fast +``` + +## Package Options + +### Google GenAI (Official) + +The official Google GenAI package provides direct access to Google's Generative AI models. This sample uses an extension method to convert the Google client to an `IChatClient`. + +> [!NOTE] +> Until PR [googleapis/dotnet-genai#81](https://github.com/googleapis/dotnet-genai/pull/81) is merged, this option requires the additional `GeminiChatClient.cs` and `GoogleGenAIExtensions.cs` files included in this sample. +> +> We appreciate any community push by liking and commenting in the above PR to get it merged and release as part of official Google GenAI package. + +### Mscc.GenerativeAI.Microsoft (Community) + +The community-driven Mscc.GenerativeAI.Microsoft package provides a ready-to-use `IChatClient` implementation for Google Gemini models through the `GeminiChatClient` class. diff --git a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_ONNX/Agent_With_ONNX.csproj b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_ONNX/Agent_With_ONNX.csproj index c4a9467179..61acc80e9c 100644 --- a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_ONNX/Agent_With_ONNX.csproj +++ b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_ONNX/Agent_With_ONNX.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable diff --git a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_ONNX/README.md b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_ONNX/README.md index cb86e0d7c4..d97b0075ac 100644 --- a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_ONNX/README.md +++ b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_ONNX/README.md @@ -4,7 +4,7 @@ WARNING: ONNX doesn't support function calling, so any function tools passed to Before you begin, ensure you have the following prerequisites: -- .NET 8.0 SDK or later +- .NET 10 SDK or later - An ONNX model downloaded to your machine You can download an ONNX model from hugging face, using git clone: diff --git a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_Ollama/Agent_With_Ollama.csproj b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_Ollama/Agent_With_Ollama.csproj index 1ad175831b..c538cbedd1 100644 --- a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_Ollama/Agent_With_Ollama.csproj +++ b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_Ollama/Agent_With_Ollama.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable diff --git a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_Ollama/README.md b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_Ollama/README.md index be76a75de0..d448f31d65 100644 --- a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_Ollama/README.md +++ b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_Ollama/README.md @@ -2,7 +2,7 @@ Before you begin, ensure you have the following prerequisites: -- .NET 8.0 SDK or later +- .NET 10 SDK or later - Docker installed and running on your machine - An Ollama model downloaded into Ollama diff --git a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_OpenAIAssistants/Agent_With_OpenAIAssistants.csproj b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_OpenAIAssistants/Agent_With_OpenAIAssistants.csproj index 0629a84bd0..eeda3eef6f 100644 --- a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_OpenAIAssistants/Agent_With_OpenAIAssistants.csproj +++ b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_OpenAIAssistants/Agent_With_OpenAIAssistants.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable diff --git a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_OpenAIAssistants/Program.cs b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_OpenAIAssistants/Program.cs index 9a91e43d37..3079bd103c 100644 --- a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_OpenAIAssistants/Program.cs +++ b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_OpenAIAssistants/Program.cs @@ -5,10 +5,13 @@ // WARNING: The Assistants API is deprecated and will be shut down. // For more information see the OpenAI documentation: https://platform.openai.com/docs/assistants/migration +#pragma warning disable CS0618 // Type or member is obsolete - OpenAI Assistants API is deprecated but still used in this sample + using Microsoft.Agents.AI; using OpenAI; +using OpenAI.Assistants; -var apiKey = Environment.GetEnvironmentVariable("OPENAI_APIKEY") ?? throw new InvalidOperationException("OPENAI_APIKEY is not set."); +var apiKey = Environment.GetEnvironmentVariable("OPENAI_API_KEY") ?? throw new InvalidOperationException("OPENAI_API_KEY is not set."); var model = Environment.GetEnvironmentVariable("OPENAI_MODEL") ?? "gpt-4o-mini"; const string JokerName = "Joker"; diff --git a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_OpenAIAssistants/README.md b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_OpenAIAssistants/README.md index 22a4bae18c..05d2380b78 100644 --- a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_OpenAIAssistants/README.md +++ b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_OpenAIAssistants/README.md @@ -5,12 +5,12 @@ For more information see the OpenAI documentation: https://platform.openai.com/d Before you begin, ensure you have the following prerequisites: -- .NET 8.0 SDK or later +- .NET 10 SDK or later - OpenAI API key Set the following environment variables: ```powershell -$env:OPENAI_APIKEY="*****" # Replace with your OpenAI API key +$env:OPENAI_API_KEY="*****" # Replace with your OpenAI API key $env:OPENAI_MODEL="gpt-4o-mini" # Optional, defaults to gpt-4o-mini ``` diff --git a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_OpenAIChatCompletion/Agent_With_OpenAIChatCompletion.csproj b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_OpenAIChatCompletion/Agent_With_OpenAIChatCompletion.csproj index 0629a84bd0..4ea7a45b8a 100644 --- a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_OpenAIChatCompletion/Agent_With_OpenAIChatCompletion.csproj +++ b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_OpenAIChatCompletion/Agent_With_OpenAIChatCompletion.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable diff --git a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_OpenAIChatCompletion/Program.cs b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_OpenAIChatCompletion/Program.cs index 9b03c989e1..b4c6d626fc 100644 --- a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_OpenAIChatCompletion/Program.cs +++ b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_OpenAIChatCompletion/Program.cs @@ -3,9 +3,11 @@ // This sample shows how to create and use a simple AI agent with OpenAI Chat Completion as the backend. using Microsoft.Agents.AI; +using Microsoft.Extensions.AI; using OpenAI; +using OpenAI.Chat; -var apiKey = Environment.GetEnvironmentVariable("OPENAI_APIKEY") ?? throw new InvalidOperationException("OPENAI_APIKEY is not set."); +var apiKey = Environment.GetEnvironmentVariable("OPENAI_API_KEY") ?? throw new InvalidOperationException("OPENAI_API_KEY is not set."); var model = Environment.GetEnvironmentVariable("OPENAI_MODEL") ?? "gpt-4o-mini"; AIAgent agent = new OpenAIClient( diff --git a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_OpenAIChatCompletion/README.md b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_OpenAIChatCompletion/README.md index 80b63e7cd0..70fc472214 100644 --- a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_OpenAIChatCompletion/README.md +++ b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_OpenAIChatCompletion/README.md @@ -2,12 +2,12 @@ Before you begin, ensure you have the following prerequisites: -- .NET 8.0 SDK or later +- .NET 10 SDK or later - OpenAI api key Set the following environment variables: ```powershell -$env:OPENAI_APIKEY="*****" # Replace with your OpenAI api key +$env:OPENAI_API_KEY="*****" # Replace with your OpenAI api key $env:OPENAI_MODEL="gpt-4o-mini" # Optional, defaults to gpt-4o-mini ``` diff --git a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_OpenAIResponses/Agent_With_OpenAIResponses.csproj b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_OpenAIResponses/Agent_With_OpenAIResponses.csproj index 0629a84bd0..eeda3eef6f 100644 --- a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_OpenAIResponses/Agent_With_OpenAIResponses.csproj +++ b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_OpenAIResponses/Agent_With_OpenAIResponses.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable diff --git a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_OpenAIResponses/Program.cs b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_OpenAIResponses/Program.cs index 1abefa0fca..df53ba8869 100644 --- a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_OpenAIResponses/Program.cs +++ b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_OpenAIResponses/Program.cs @@ -4,8 +4,9 @@ using Microsoft.Agents.AI; using OpenAI; +using OpenAI.Responses; -var apiKey = Environment.GetEnvironmentVariable("OPENAI_APIKEY") ?? throw new InvalidOperationException("OPENAI_APIKEY is not set."); +var apiKey = Environment.GetEnvironmentVariable("OPENAI_API_KEY") ?? throw new InvalidOperationException("OPENAI_API_KEY is not set."); var model = Environment.GetEnvironmentVariable("OPENAI_MODEL") ?? "gpt-4o-mini"; AIAgent agent = new OpenAIClient( diff --git a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_OpenAIResponses/README.md b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_OpenAIResponses/README.md index 80b63e7cd0..70fc472214 100644 --- a/dotnet/samples/GettingStarted/AgentProviders/Agent_With_OpenAIResponses/README.md +++ b/dotnet/samples/GettingStarted/AgentProviders/Agent_With_OpenAIResponses/README.md @@ -2,12 +2,12 @@ Before you begin, ensure you have the following prerequisites: -- .NET 8.0 SDK or later +- .NET 10 SDK or later - OpenAI api key Set the following environment variables: ```powershell -$env:OPENAI_APIKEY="*****" # Replace with your OpenAI api key +$env:OPENAI_API_KEY="*****" # Replace with your OpenAI api key $env:OPENAI_MODEL="gpt-4o-mini" # Optional, defaults to gpt-4o-mini ``` diff --git a/dotnet/samples/GettingStarted/AgentProviders/README.md b/dotnet/samples/GettingStarted/AgentProviders/README.md index 4e84cd4f08..964e560c9a 100644 --- a/dotnet/samples/GettingStarted/AgentProviders/README.md +++ b/dotnet/samples/GettingStarted/AgentProviders/README.md @@ -15,7 +15,9 @@ See the README.md for each sample for the prerequisites for that sample. |Sample|Description| |---|---| |[Creating an AIAgent with A2A](./Agent_With_A2A/)|This sample demonstrates how to create AIAgent for an existing A2A agent.| -|[Creating an AIAgent with AzureFoundry Agent](./Agent_With_AzureFoundryAgent/)|This sample demonstrates how to create an Azure Foundry agent and expose it as an AIAgent| +|[Creating an AIAgent with Anthropic](./Agent_With_Anthropic/)|This sample demonstrates how to create an AIAgent using Anthropic Claude models as the underlying inference service| +|[Creating an AIAgent with Foundry Agents using Azure.AI.Agents.Persistent](./Agent_With_AzureAIAgentsPersistent/)|This sample demonstrates how to create a Foundry Persistent agent and expose it as an AIAgent using the Azure.AI.Agents.Persistent SDK| +|[Creating an AIAgent with Foundry Agents using Azure.AI.Project](./Agent_With_AzureAIProject/)|This sample demonstrates how to create an Foundry Project agent and expose it as an AIAgent using the Azure.AI.Project SDK| |[Creating an AIAgent with AzureFoundry Model](./Agent_With_AzureFoundryModel/)|This sample demonstrates how to use any model deployed to Azure Foundry to create an AIAgent| |[Creating an AIAgent with Azure OpenAI ChatCompletion](./Agent_With_AzureOpenAIChatCompletion/)|This sample demonstrates how to create an AIAgent using Azure OpenAI ChatCompletion as the underlying inference service| |[Creating an AIAgent with Azure OpenAI Responses](./Agent_With_AzureOpenAIResponses/)|This sample demonstrates how to create an AIAgent using Azure OpenAI Responses as the underlying inference service| diff --git a/dotnet/samples/GettingStarted/AgentWithAnthropic/Agent_Anthropic_Step01_Running/Agent_Anthropic_Step01_Running.csproj b/dotnet/samples/GettingStarted/AgentWithAnthropic/Agent_Anthropic_Step01_Running/Agent_Anthropic_Step01_Running.csproj new file mode 100644 index 0000000000..09359c5e78 --- /dev/null +++ b/dotnet/samples/GettingStarted/AgentWithAnthropic/Agent_Anthropic_Step01_Running/Agent_Anthropic_Step01_Running.csproj @@ -0,0 +1,15 @@ + + + + Exe + net10.0 + + enable + enable + + + + + + + diff --git a/dotnet/samples/GettingStarted/AgentWithAnthropic/Agent_Anthropic_Step01_Running/Program.cs b/dotnet/samples/GettingStarted/AgentWithAnthropic/Agent_Anthropic_Step01_Running/Program.cs new file mode 100644 index 0000000000..cf7e29c2fe --- /dev/null +++ b/dotnet/samples/GettingStarted/AgentWithAnthropic/Agent_Anthropic_Step01_Running/Program.cs @@ -0,0 +1,24 @@ +// Copyright (c) Microsoft. All rights reserved. + +// This sample shows how to create and use a simple AI agent with Anthropic as the backend. + +using Anthropic; +using Anthropic.Core; +using Microsoft.Agents.AI; +using Microsoft.Extensions.AI; + +var apiKey = Environment.GetEnvironmentVariable("ANTHROPIC_API_KEY") ?? throw new InvalidOperationException("ANTHROPIC_API_KEY is not set."); +var model = Environment.GetEnvironmentVariable("ANTHROPIC_MODEL") ?? "claude-haiku-4-5"; + +AIAgent agent = new AnthropicClient(new ClientOptions { APIKey = apiKey }) + .CreateAIAgent(model: model, instructions: "You are good at telling jokes.", name: "Joker"); + +// Invoke the agent and output the text result. +var response = await agent.RunAsync("Tell me a joke about a pirate."); +Console.WriteLine(response); + +// Invoke the agent with streaming support. +await foreach (var update in agent.RunStreamingAsync("Tell me a joke about a pirate.")) +{ + Console.WriteLine(update); +} diff --git a/dotnet/samples/GettingStarted/AgentWithAnthropic/Agent_Anthropic_Step01_Running/README.md b/dotnet/samples/GettingStarted/AgentWithAnthropic/Agent_Anthropic_Step01_Running/README.md new file mode 100644 index 0000000000..4800650bd9 --- /dev/null +++ b/dotnet/samples/GettingStarted/AgentWithAnthropic/Agent_Anthropic_Step01_Running/README.md @@ -0,0 +1,43 @@ +# Running a simple agent with Anthropic + +This sample demonstrates how to create and run a basic agent with Anthropic Claude models. + +## What this sample demonstrates + +- Creating an AI agent with Anthropic Claude +- Running a simple agent with instructions +- Managing agent lifecycle + +## Prerequisites + +Before you begin, ensure you have the following prerequisites: + +- .NET 8.0 SDK or later +- Anthropic API key configured + +**Note**: This sample uses Anthropic Claude models. For more information, see [Anthropic documentation](https://docs.anthropic.com/). + +Set the following environment variables: + +```powershell +$env:ANTHROPIC_API_KEY="your-anthropic-api-key" # Replace with your Anthropic API key +$env:ANTHROPIC_MODEL="your-anthropic-model" # Replace with your Anthropic model +``` + +## Run the sample + +Navigate to the AgentWithAnthropic sample directory and run: + +```powershell +cd dotnet\samples\GettingStarted\AgentWithAnthropic +dotnet run --project .\Agent_Anthropic_Step01_Running +``` + +## Expected behavior + +The sample will: + +1. Create an agent with Anthropic Claude +2. Run the agent with a simple prompt +3. Display the agent's response + diff --git a/dotnet/samples/GettingStarted/AgentWithAnthropic/Agent_Anthropic_Step02_Reasoning/Agent_Anthropic_Step02_Reasoning.csproj b/dotnet/samples/GettingStarted/AgentWithAnthropic/Agent_Anthropic_Step02_Reasoning/Agent_Anthropic_Step02_Reasoning.csproj new file mode 100644 index 0000000000..fc0914f1fc --- /dev/null +++ b/dotnet/samples/GettingStarted/AgentWithAnthropic/Agent_Anthropic_Step02_Reasoning/Agent_Anthropic_Step02_Reasoning.csproj @@ -0,0 +1,15 @@ + + + + Exe + net10.0 + + enable + enable + + + + + + + \ No newline at end of file diff --git a/dotnet/samples/GettingStarted/AgentWithAnthropic/Agent_Anthropic_Step02_Reasoning/Program.cs b/dotnet/samples/GettingStarted/AgentWithAnthropic/Agent_Anthropic_Step02_Reasoning/Program.cs new file mode 100644 index 0000000000..d362a9dd0d --- /dev/null +++ b/dotnet/samples/GettingStarted/AgentWithAnthropic/Agent_Anthropic_Step02_Reasoning/Program.cs @@ -0,0 +1,59 @@ +// Copyright (c) Microsoft. All rights reserved. + +// This sample shows how to create and use an AI agent with reasoning capabilities. + +using Anthropic; +using Anthropic.Core; +using Anthropic.Models.Messages; +using Microsoft.Agents.AI; +using Microsoft.Extensions.AI; + +var apiKey = Environment.GetEnvironmentVariable("ANTHROPIC_API_KEY") ?? throw new InvalidOperationException("ANTHROPIC_API_KEY is not set."); +var model = Environment.GetEnvironmentVariable("ANTHROPIC_MODEL") ?? "claude-haiku-4-5"; +var maxTokens = 4096; +var thinkingTokens = 2048; + +var agent = new AnthropicClient(new ClientOptions { APIKey = apiKey }) + .CreateAIAgent( + model: model, + clientFactory: (chatClient) => chatClient + .AsBuilder() + .ConfigureOptions( + options => options.RawRepresentationFactory = (_) => new MessageCreateParams() + { + Model = options.ModelId ?? model, + MaxTokens = options.MaxOutputTokens ?? maxTokens, + Messages = [], + Thinking = new ThinkingConfigParam(new ThinkingConfigEnabled(budgetTokens: thinkingTokens)) + }) + .Build()); + +Console.WriteLine("1. Non-streaming:"); +var response = await agent.RunAsync("Solve this problem step by step: If a train travels 60 miles per hour and needs to cover 180 miles, how long will the journey take? Show your reasoning."); + +Console.WriteLine("#### Start Thinking ####"); +Console.WriteLine($"\e[92m{string.Join("\n", response.Messages.SelectMany(m => m.Contents.OfType().Select(c => c.Text)))}\e[0m"); +Console.WriteLine("#### End Thinking ####"); + +Console.WriteLine("\n#### Final Answer ####"); +Console.WriteLine(response.Text); + +Console.WriteLine("Token usage:"); +Console.WriteLine($"Input: {response.Usage?.InputTokenCount}, Output: {response.Usage?.OutputTokenCount}, {string.Join(", ", response.Usage?.AdditionalCounts ?? [])}"); +Console.WriteLine(); + +Console.WriteLine("2. Streaming"); +await foreach (var update in agent.RunStreamingAsync("Explain the theory of relativity in simple terms.")) +{ + foreach (var item in update.Contents) + { + if (item is TextReasoningContent reasoningContent) + { + Console.WriteLine($"\e[92m{reasoningContent.Text}\e[0m"); + } + else if (item is TextContent textContent) + { + Console.WriteLine(textContent.Text); + } + } +} diff --git a/dotnet/samples/GettingStarted/AgentWithAnthropic/Agent_Anthropic_Step02_Reasoning/README.md b/dotnet/samples/GettingStarted/AgentWithAnthropic/Agent_Anthropic_Step02_Reasoning/README.md new file mode 100644 index 0000000000..ae088b2386 --- /dev/null +++ b/dotnet/samples/GettingStarted/AgentWithAnthropic/Agent_Anthropic_Step02_Reasoning/README.md @@ -0,0 +1,46 @@ +# Using reasoning with Anthropic agents + +This sample demonstrates how to use extended thinking/reasoning capabilities with Anthropic Claude agents. + +## What this sample demonstrates + +- Creating an AI agent with Anthropic Claude extended thinking +- Using reasoning capabilities for complex problem solving +- Extracting thinking and response content from agent output +- Managing agent lifecycle + +## Prerequisites + +Before you begin, ensure you have the following prerequisites: + +- .NET 8.0 SDK or later +- Anthropic API key configured +- Access to Anthropic Claude models with extended thinking support + +**Note**: This sample uses Anthropic Claude models with extended thinking. For more information, see [Anthropic documentation](https://docs.anthropic.com/). + +Set the following environment variables: + +```powershell +$env:ANTHROPIC_API_KEY="your-anthropic-api-key" # Replace with your Anthropic API key +$env:ANTHROPIC_MODEL="your-anthropic-model" # Replace with your Anthropic model +``` + +## Run the sample + +Navigate to the AgentWithAnthropic sample directory and run: + +```powershell +cd dotnet\samples\GettingStarted\AgentWithAnthropic +dotnet run --project .\Agent_Anthropic_Step02_Reasoning +``` + +## Expected behavior + +The sample will: + +1. Create an agent with Anthropic Claude extended thinking enabled +2. Run the agent with a complex reasoning prompt +3. Display the agent's thinking process +4. Display the agent's final response + diff --git a/dotnet/samples/GettingStarted/AgentWithAnthropic/Agent_Anthropic_Step03_UsingFunctionTools/Agent_Anthropic_Step03_UsingFunctionTools.csproj b/dotnet/samples/GettingStarted/AgentWithAnthropic/Agent_Anthropic_Step03_UsingFunctionTools/Agent_Anthropic_Step03_UsingFunctionTools.csproj new file mode 100644 index 0000000000..fdb9a2f50f --- /dev/null +++ b/dotnet/samples/GettingStarted/AgentWithAnthropic/Agent_Anthropic_Step03_UsingFunctionTools/Agent_Anthropic_Step03_UsingFunctionTools.csproj @@ -0,0 +1,15 @@ + + + + Exe + net10.0 + + enable + enable + + + + + + + diff --git a/dotnet/samples/GettingStarted/AgentWithAnthropic/Agent_Anthropic_Step03_UsingFunctionTools/Program.cs b/dotnet/samples/GettingStarted/AgentWithAnthropic/Agent_Anthropic_Step03_UsingFunctionTools/Program.cs new file mode 100644 index 0000000000..a56db8d4a2 --- /dev/null +++ b/dotnet/samples/GettingStarted/AgentWithAnthropic/Agent_Anthropic_Step03_UsingFunctionTools/Program.cs @@ -0,0 +1,37 @@ +// Copyright (c) Microsoft. All rights reserved. + +// This sample demonstrates how to use an agent with function tools. +// It shows both non-streaming and streaming agent interactions using weather-related tools. + +using System.ComponentModel; +using Anthropic; +using Microsoft.Agents.AI; +using Microsoft.Extensions.AI; + +var apiKey = Environment.GetEnvironmentVariable("ANTHROPIC_API_KEY") ?? throw new InvalidOperationException("ANTHROPIC_API_KEY is not set."); +var model = Environment.GetEnvironmentVariable("ANTHROPIC_MODEL") ?? "claude-haiku-4-5"; + +[Description("Get the weather for a given location.")] +static string GetWeather([Description("The location to get the weather for.")] string location) + => $"The weather in {location} is cloudy with a high of 15°C."; + +const string AssistantInstructions = "You are a helpful assistant that can get weather information."; +const string AssistantName = "WeatherAssistant"; + +// Define the agent with function tools. +AITool tool = AIFunctionFactory.Create(GetWeather); + +// Get anthropic client to create agents. +AIAgent agent = new AnthropicClient { APIKey = apiKey } + .CreateAIAgent(model: model, instructions: AssistantInstructions, name: AssistantName, tools: [tool]); + +// Non-streaming agent interaction with function tools. +AgentThread thread = agent.GetNewThread(); +Console.WriteLine(await agent.RunAsync("What is the weather like in Amsterdam?", thread)); + +// Streaming agent interaction with function tools. +thread = agent.GetNewThread(); +await foreach (AgentRunResponseUpdate update in agent.RunStreamingAsync("What is the weather like in Amsterdam?", thread)) +{ + Console.WriteLine(update); +} diff --git a/dotnet/samples/GettingStarted/AgentWithAnthropic/Agent_Anthropic_Step03_UsingFunctionTools/README.md b/dotnet/samples/GettingStarted/AgentWithAnthropic/Agent_Anthropic_Step03_UsingFunctionTools/README.md new file mode 100644 index 0000000000..6c905864ef --- /dev/null +++ b/dotnet/samples/GettingStarted/AgentWithAnthropic/Agent_Anthropic_Step03_UsingFunctionTools/README.md @@ -0,0 +1,47 @@ +# Using Function Tools with Anthropic agents + +This sample demonstrates how to use function tools with Anthropic Claude agents, allowing agents to call custom functions to retrieve information. + +## What this sample demonstrates + +- Creating function tools using AIFunctionFactory +- Passing function tools to an Anthropic Claude agent +- Running agents with function tools (text output) +- Running agents with function tools (streaming output) +- Managing agent lifecycle + +## Prerequisites + +Before you begin, ensure you have the following prerequisites: + +- .NET 8.0 SDK or later +- Anthropic API key configured + +**Note**: This sample uses Anthropic Claude models. For more information, see [Anthropic documentation](https://docs.anthropic.com/). + +Set the following environment variables: + +```powershell +$env:ANTHROPIC_API_KEY="your-anthropic-api-key" # Replace with your Anthropic API key +$env:ANTHROPIC_MODEL="your-anthropic-model" # Replace with your Anthropic model +``` + +## Run the sample + +Navigate to the AgentWithAnthropic sample directory and run: + +```powershell +cd dotnet\samples\GettingStarted\AgentWithAnthropic +dotnet run --project .\Agent_Anthropic_Step03_UsingFunctionTools +``` + +## Expected behavior + +The sample will: + +1. Create an agent named "WeatherAssistant" with a GetWeather function tool +2. Run the agent with a text prompt asking about weather +3. The agent will invoke the GetWeather function tool to retrieve weather information +4. Run the agent again with streaming to display the response as it's generated +5. Clean up resources by deleting the agent + diff --git a/dotnet/samples/GettingStarted/AgentWithAnthropic/README.md b/dotnet/samples/GettingStarted/AgentWithAnthropic/README.md new file mode 100644 index 0000000000..44c15b384b --- /dev/null +++ b/dotnet/samples/GettingStarted/AgentWithAnthropic/README.md @@ -0,0 +1,72 @@ +# Getting started with agents using Anthropic + +The getting started with agents using Anthropic samples demonstrate the fundamental concepts and functionalities +of single agents using Anthropic as the AI provider. + +These samples use Anthropic Claude models as the AI provider and use ChatCompletion as the type of service. + +For other samples that demonstrate how to create and configure each type of agent that come with the agent framework, +see the [How to create an agent for each provider](../AgentProviders/README.md) samples. + +## Getting started with agents using Anthropic prerequisites + +Before you begin, ensure you have the following prerequisites: + +- .NET 8.0 SDK or later +- Anthropic API key configured +- User has access to Anthropic Claude models + +**Note**: These samples use Anthropic Claude models. For more information, see [Anthropic documentation](https://docs.anthropic.com/). + +## Using Anthropic with Azure Foundry + +To use Anthropic with Azure Foundry, you can check the sample [AgentProviders/Agent_With_Anthropic](../AgentProviders/Agent_With_Anthropic/README.md) for more details. + +## Samples + +|Sample|Description| +|---|---| +|[Running a simple agent](./Agent_Anthropic_Step01_Running/)|This sample demonstrates how to create and run a basic agent with Anthropic Claude| +|[Using reasoning with an agent](./Agent_Anthropic_Step02_Reasoning/)|This sample demonstrates how to use extended thinking/reasoning capabilities with Anthropic Claude agents| +|[Using function tools with an agent](./Agent_Anthropic_Step03_UsingFunctionTools/)|This sample demonstrates how to use function tools with an Anthropic Claude agent| + +## Running the samples from the console + +To run the samples, navigate to the desired sample directory, e.g. + +```powershell +cd Agent_Anthropic_Step01_Running +``` + +Set the following environment variables: + +```powershell +$env:ANTHROPIC_API_KEY="your-anthropic-api-key" # Replace with your Anthropic API key +``` + +If the variables are not set, you will be prompted for the values when running the samples. + +Execute the following command to build the sample: + +```powershell +dotnet build +``` + +Execute the following command to run the sample: + +```powershell +dotnet run --no-build +``` + +Or just build and run in one step: + +```powershell +dotnet run +``` + +## Running the samples from Visual Studio + +Open the solution in Visual Studio and set the desired sample project as the startup project. Then, run the project using the built-in debugger or by pressing `F5`. + +You will be prompted for any required environment variables if they are not already set. + diff --git a/dotnet/samples/GettingStarted/AgentWithMemory/AgentWithMemory_Step01_ChatHistoryMemory/AgentWithMemory_Step01_ChatHistoryMemory.csproj b/dotnet/samples/GettingStarted/AgentWithMemory/AgentWithMemory_Step01_ChatHistoryMemory/AgentWithMemory_Step01_ChatHistoryMemory.csproj new file mode 100644 index 0000000000..860089b621 --- /dev/null +++ b/dotnet/samples/GettingStarted/AgentWithMemory/AgentWithMemory_Step01_ChatHistoryMemory/AgentWithMemory_Step01_ChatHistoryMemory.csproj @@ -0,0 +1,22 @@ + + + + Exe + net10.0 + + enable + enable + + + + + + + + + + + + + + diff --git a/dotnet/samples/GettingStarted/AgentWithMemory/AgentWithMemory_Step01_ChatHistoryMemory/Program.cs b/dotnet/samples/GettingStarted/AgentWithMemory/AgentWithMemory_Step01_ChatHistoryMemory/Program.cs new file mode 100644 index 0000000000..a11edafabc --- /dev/null +++ b/dotnet/samples/GettingStarted/AgentWithMemory/AgentWithMemory_Step01_ChatHistoryMemory/Program.cs @@ -0,0 +1,60 @@ +// Copyright (c) Microsoft. All rights reserved. + +// This sample shows how to create and use a simple AI agent that stores chat messages in a vector store using the ChatHistoryMemoryProvider. +// It can then use the chat history from prior conversations to inform responses in new conversations. + +using Azure.AI.OpenAI; +using Azure.Identity; +using Microsoft.Agents.AI; +using Microsoft.Extensions.AI; +using Microsoft.Extensions.VectorData; +using Microsoft.SemanticKernel.Connectors.InMemory; +using OpenAI.Chat; + +var endpoint = Environment.GetEnvironmentVariable("AZURE_OPENAI_ENDPOINT") ?? throw new InvalidOperationException("AZURE_OPENAI_ENDPOINT is not set."); +var deploymentName = Environment.GetEnvironmentVariable("AZURE_OPENAI_DEPLOYMENT_NAME") ?? "gpt-4o-mini"; +var embeddingDeploymentName = Environment.GetEnvironmentVariable("AZURE_OPENAI_EMBEDDING_DEPLOYMENT_NAME") ?? "text-embedding-3-large"; + +// Create a vector store to store the chat messages in. +// For demonstration purposes, we are using an in-memory vector store. +// Replace this with a vector store implementation of your choice that can persist the chat history long term. +VectorStore vectorStore = new InMemoryVectorStore(new InMemoryVectorStoreOptions() +{ + EmbeddingGenerator = new AzureOpenAIClient(new Uri(endpoint), new AzureCliCredential()) + .GetEmbeddingClient(embeddingDeploymentName) + .AsIEmbeddingGenerator() +}); + +// Create the agent and add the ChatHistoryMemoryProvider to store chat messages in the vector store. +AIAgent agent = new AzureOpenAIClient( + new Uri(endpoint), + new AzureCliCredential()) + .GetChatClient(deploymentName) + .CreateAIAgent(new ChatClientAgentOptions + { + ChatOptions = new() { Instructions = "You are good at telling jokes." }, + Name = "Joker", + AIContextProviderFactory = (ctx) => new ChatHistoryMemoryProvider( + vectorStore, + collectionName: "chathistory", + vectorDimensions: 3072, + // Configure the scope values under which chat messages will be stored. + // In this case, we are using a fixed user ID and a unique thread ID for each new thread. + storageScope: new() { UserId = "UID1", ThreadId = new Guid().ToString() }, + // Configure the scope which would be used to search for relevant prior messages. + // In this case, we are searching for any messages for the user across all threads. + searchScope: new() { UserId = "UID1" }) + }); + +// Start a new thread for the agent conversation. +AgentThread thread = agent.GetNewThread(); + +// Run the agent with the thread that stores conversation history in the vector store. +Console.WriteLine(await agent.RunAsync("I like jokes about Pirates. Tell me a joke about a pirate.", thread)); + +// Start a second thread. Since we configured the search scope to be across all threads for the user, +// the agent should remember that the user likes pirate jokes. +AgentThread thread2 = agent.GetNewThread(); + +// Run the agent with the second thread. +Console.WriteLine(await agent.RunAsync("Tell me a joke that I might like.", thread2)); diff --git a/dotnet/samples/GettingStarted/Agents/Agent_Step19_Mem0Provider/Agent_Step19_Mem0Provider.csproj b/dotnet/samples/GettingStarted/AgentWithMemory/AgentWithMemory_Step02_MemoryUsingMem0/AgentWithMemory_Step02_MemoryUsingMem0.csproj similarity index 92% rename from dotnet/samples/GettingStarted/Agents/Agent_Step19_Mem0Provider/Agent_Step19_Mem0Provider.csproj rename to dotnet/samples/GettingStarted/AgentWithMemory/AgentWithMemory_Step02_MemoryUsingMem0/AgentWithMemory_Step02_MemoryUsingMem0.csproj index 9d7aa41a99..1e0863d66f 100644 --- a/dotnet/samples/GettingStarted/Agents/Agent_Step19_Mem0Provider/Agent_Step19_Mem0Provider.csproj +++ b/dotnet/samples/GettingStarted/AgentWithMemory/AgentWithMemory_Step02_MemoryUsingMem0/AgentWithMemory_Step02_MemoryUsingMem0.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable diff --git a/dotnet/samples/GettingStarted/Agents/Agent_Step19_Mem0Provider/Program.cs b/dotnet/samples/GettingStarted/AgentWithMemory/AgentWithMemory_Step02_MemoryUsingMem0/Program.cs similarity index 93% rename from dotnet/samples/GettingStarted/Agents/Agent_Step19_Mem0Provider/Program.cs rename to dotnet/samples/GettingStarted/AgentWithMemory/AgentWithMemory_Step02_MemoryUsingMem0/Program.cs index 539ebbaecb..739c5e3f13 100644 --- a/dotnet/samples/GettingStarted/Agents/Agent_Step19_Mem0Provider/Program.cs +++ b/dotnet/samples/GettingStarted/AgentWithMemory/AgentWithMemory_Step02_MemoryUsingMem0/Program.cs @@ -11,7 +11,7 @@ using Microsoft.Agents.AI; using Microsoft.Agents.AI.Mem0; using Microsoft.Extensions.AI; -using OpenAI; +using OpenAI.Chat; var endpoint = Environment.GetEnvironmentVariable("AZURE_OPENAI_ENDPOINT") ?? throw new InvalidOperationException("AZURE_OPENAI_ENDPOINT is not set."); var deploymentName = Environment.GetEnvironmentVariable("AZURE_OPENAI_DEPLOYMENT_NAME") ?? "gpt-4o-mini"; @@ -30,8 +30,8 @@ .GetChatClient(deploymentName) .CreateAIAgent(new ChatClientAgentOptions() { - Instructions = "You are a friendly travel assistant. Use known memories about the user when responding, and do not invent details.", - AIContextProviderFactory = ctx => ctx.SerializedState.ValueKind is not JsonValueKind.Null or JsonValueKind.Undefined + ChatOptions = new() { Instructions = "You are a friendly travel assistant. Use known memories about the user when responding, and do not invent details." }, + AIContextProviderFactory = ctx => ctx.SerializedState.ValueKind is not JsonValueKind.Null and not JsonValueKind.Undefined // If each thread should have its own Mem0 scope, you can create a new id per thread here: // ? new Mem0Provider(mem0HttpClient, new Mem0ProviderScope() { ThreadId = Guid.NewGuid().ToString() }) // In this case we are storing memories scoped by application and user instead so that memories are retained across threads. diff --git a/dotnet/samples/GettingStarted/Agents/Agent_Step18_TextSearchRag/Agent_Step18_TextSearchRag.csproj b/dotnet/samples/GettingStarted/AgentWithMemory/AgentWithMemory_Step03_CustomMemory/AgentWithMemory_Step03_CustomMemory.csproj similarity index 91% rename from dotnet/samples/GettingStarted/Agents/Agent_Step18_TextSearchRag/Agent_Step18_TextSearchRag.csproj rename to dotnet/samples/GettingStarted/AgentWithMemory/AgentWithMemory_Step03_CustomMemory/AgentWithMemory_Step03_CustomMemory.csproj index 8298cfe6e8..0f9de7c359 100644 --- a/dotnet/samples/GettingStarted/Agents/Agent_Step18_TextSearchRag/Agent_Step18_TextSearchRag.csproj +++ b/dotnet/samples/GettingStarted/AgentWithMemory/AgentWithMemory_Step03_CustomMemory/AgentWithMemory_Step03_CustomMemory.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable diff --git a/dotnet/samples/GettingStarted/Agents/Agent_Step13_Memory/Program.cs b/dotnet/samples/GettingStarted/AgentWithMemory/AgentWithMemory_Step03_CustomMemory/Program.cs similarity index 98% rename from dotnet/samples/GettingStarted/Agents/Agent_Step13_Memory/Program.cs rename to dotnet/samples/GettingStarted/AgentWithMemory/AgentWithMemory_Step03_CustomMemory/Program.cs index ad59deb97f..5727e8ca3c 100644 --- a/dotnet/samples/GettingStarted/Agents/Agent_Step13_Memory/Program.cs +++ b/dotnet/samples/GettingStarted/AgentWithMemory/AgentWithMemory_Step03_CustomMemory/Program.cs @@ -12,7 +12,6 @@ using Azure.Identity; using Microsoft.Agents.AI; using Microsoft.Extensions.AI; -using OpenAI; using OpenAI.Chat; using SampleApp; @@ -33,7 +32,7 @@ // and its storage to that user id. AIAgent agent = chatClient.CreateAIAgent(new ChatClientAgentOptions() { - Instructions = "You are a friendly assistant. Always address the user by their name.", + ChatOptions = new() { Instructions = "You are a friendly assistant. Always address the user by their name." }, AIContextProviderFactory = ctx => new UserInfoMemory(chatClient.AsIChatClient(), ctx.SerializedState, ctx.JsonSerializerOptions) }); diff --git a/dotnet/samples/GettingStarted/AgentWithMemory/README.md b/dotnet/samples/GettingStarted/AgentWithMemory/README.md new file mode 100644 index 0000000000..903fcf1b78 --- /dev/null +++ b/dotnet/samples/GettingStarted/AgentWithMemory/README.md @@ -0,0 +1,9 @@ +# Agent Framework Retrieval Augmented Generation (RAG) + +These samples show how to create an agent with the Agent Framework that uses Memory to remember previous conversations or facts from previous conversations. + +|Sample|Description| +|---|---| +|[Chat History memory](./AgentWithMemory_Step01_ChatHistoryMemory/)|This sample demonstrates how to enable an agent to remember messages from previous conversations.| +|[Memory with MemoryStore](./AgentWithMemory_Step02_MemoryUsingMem0/)|This sample demonstrates how to create and run an agent that uses the Mem0 service to extract and retrieve individual memories.| +|[Custom Memory Implementation](./AgentWithMemory_Step03_CustomMemory/)|This sample demonstrates how to create a custom memory component and attach it to an agent.| diff --git a/dotnet/samples/GettingStarted/AgentWithOpenAI/Agent_OpenAI_Step01_Running/Agent_OpenAI_Step01_Running.csproj b/dotnet/samples/GettingStarted/AgentWithOpenAI/Agent_OpenAI_Step01_Running/Agent_OpenAI_Step01_Running.csproj index 0629a84bd0..eeda3eef6f 100644 --- a/dotnet/samples/GettingStarted/AgentWithOpenAI/Agent_OpenAI_Step01_Running/Agent_OpenAI_Step01_Running.csproj +++ b/dotnet/samples/GettingStarted/AgentWithOpenAI/Agent_OpenAI_Step01_Running/Agent_OpenAI_Step01_Running.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable diff --git a/dotnet/samples/GettingStarted/AgentWithOpenAI/Agent_OpenAI_Step02_Reasoning/Agent_OpenAI_Step02_Reasoning.csproj b/dotnet/samples/GettingStarted/AgentWithOpenAI/Agent_OpenAI_Step02_Reasoning/Agent_OpenAI_Step02_Reasoning.csproj index 4253d9cf9e..78f0981676 100644 --- a/dotnet/samples/GettingStarted/AgentWithOpenAI/Agent_OpenAI_Step02_Reasoning/Agent_OpenAI_Step02_Reasoning.csproj +++ b/dotnet/samples/GettingStarted/AgentWithOpenAI/Agent_OpenAI_Step02_Reasoning/Agent_OpenAI_Step02_Reasoning.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable diff --git a/dotnet/samples/GettingStarted/AgentWithOpenAI/Agent_OpenAI_Step02_Reasoning/Program.cs b/dotnet/samples/GettingStarted/AgentWithOpenAI/Agent_OpenAI_Step02_Reasoning/Program.cs index 01f8d46ee1..e06a8cc76f 100644 --- a/dotnet/samples/GettingStarted/AgentWithOpenAI/Agent_OpenAI_Step02_Reasoning/Program.cs +++ b/dotnet/samples/GettingStarted/AgentWithOpenAI/Agent_OpenAI_Step02_Reasoning/Program.cs @@ -7,7 +7,7 @@ using OpenAI; using OpenAI.Responses; -var apiKey = Environment.GetEnvironmentVariable("OPENAI_APIKEY") ?? throw new InvalidOperationException("OPENAI_APIKEY is not set."); +var apiKey = Environment.GetEnvironmentVariable("OPENAI_API_KEY") ?? throw new InvalidOperationException("OPENAI_API_KEY is not set."); var model = Environment.GetEnvironmentVariable("OPENAI_MODEL") ?? "gpt-5"; var client = new OpenAIClient(apiKey) diff --git a/dotnet/samples/GettingStarted/AgentWithOpenAI/Agent_OpenAI_Step03_CreateFromChatClient/Agent_OpenAI_Step03_CreateFromChatClient.csproj b/dotnet/samples/GettingStarted/AgentWithOpenAI/Agent_OpenAI_Step03_CreateFromChatClient/Agent_OpenAI_Step03_CreateFromChatClient.csproj new file mode 100644 index 0000000000..eeda3eef6f --- /dev/null +++ b/dotnet/samples/GettingStarted/AgentWithOpenAI/Agent_OpenAI_Step03_CreateFromChatClient/Agent_OpenAI_Step03_CreateFromChatClient.csproj @@ -0,0 +1,15 @@ + + + + Exe + net10.0 + + enable + enable + + + + + + + diff --git a/dotnet/src/Microsoft.Agents.AI.OpenAI/OpenAIChatClientAgent.cs b/dotnet/samples/GettingStarted/AgentWithOpenAI/Agent_OpenAI_Step03_CreateFromChatClient/OpenAIChatClientAgent.cs similarity index 95% rename from dotnet/src/Microsoft.Agents.AI.OpenAI/OpenAIChatClientAgent.cs rename to dotnet/samples/GettingStarted/AgentWithOpenAI/Agent_OpenAI_Step03_CreateFromChatClient/OpenAIChatClientAgent.cs index b529e1151b..b295bfecea 100644 --- a/dotnet/src/Microsoft.Agents.AI.OpenAI/OpenAIChatClientAgent.cs +++ b/dotnet/samples/GettingStarted/AgentWithOpenAI/Agent_OpenAI_Step03_CreateFromChatClient/OpenAIChatClientAgent.cs @@ -3,11 +3,10 @@ using Microsoft.Agents.AI; using Microsoft.Extensions.AI; using Microsoft.Extensions.Logging; -using Microsoft.Shared.Diagnostics; using OpenAI.Chat; using ChatMessage = OpenAI.Chat.ChatMessage; -namespace OpenAI; +namespace OpenAIChatClientSample; /// /// Provides an backed by an OpenAI chat completion implementation. @@ -32,7 +31,7 @@ public OpenAIChatClientAgent( { Name = name, Description = description, - Instructions = instructions, + ChatOptions = new ChatOptions() { Instructions = instructions }, }, loggerFactory) { } @@ -45,7 +44,7 @@ public OpenAIChatClientAgent( /// Optional instance of public OpenAIChatClientAgent( ChatClient client, ChatClientAgentOptions options, ILoggerFactory? loggerFactory = null) : - base(new ChatClientAgent(Throw.IfNull(client).AsIChatClient(), options, loggerFactory)) + base(new ChatClientAgent((client ?? throw new ArgumentNullException(nameof(client))).AsIChatClient(), options, loggerFactory)) { } diff --git a/dotnet/samples/GettingStarted/AgentWithOpenAI/Agent_OpenAI_Step03_CreateFromChatClient/Program.cs b/dotnet/samples/GettingStarted/AgentWithOpenAI/Agent_OpenAI_Step03_CreateFromChatClient/Program.cs new file mode 100644 index 0000000000..b046afbf0d --- /dev/null +++ b/dotnet/samples/GettingStarted/AgentWithOpenAI/Agent_OpenAI_Step03_CreateFromChatClient/Program.cs @@ -0,0 +1,32 @@ +// Copyright (c) Microsoft. All rights reserved. + +// This sample demonstrates how to create an AI agent directly from an OpenAI.Chat.ChatClient instance using OpenAIChatClientAgent. + +using OpenAI; +using OpenAI.Chat; +using OpenAIChatClientSample; + +string apiKey = Environment.GetEnvironmentVariable("OPENAI_API_KEY") ?? throw new InvalidOperationException("OPENAI_API_KEY is not set."); +string model = Environment.GetEnvironmentVariable("OPENAI_MODEL") ?? "gpt-4o-mini"; + +// Create a ChatClient directly from OpenAIClient +ChatClient chatClient = new OpenAIClient(apiKey).GetChatClient(model); + +// Create an agent directly from the ChatClient using OpenAIChatClientAgent +OpenAIChatClientAgent agent = new(chatClient, instructions: "You are good at telling jokes.", name: "Joker"); + +UserChatMessage chatMessage = new("Tell me a joke about a pirate."); + +// Invoke the agent and output the text result. +ChatCompletion chatCompletion = await agent.RunAsync([chatMessage]); +Console.WriteLine(chatCompletion.Content.Last().Text); + +// Invoke the agent with streaming support. +IAsyncEnumerable completionUpdates = agent.RunStreamingAsync([chatMessage]); +await foreach (StreamingChatCompletionUpdate completionUpdate in completionUpdates) +{ + if (completionUpdate.ContentUpdate.Count > 0) + { + Console.WriteLine(completionUpdate.ContentUpdate[0].Text); + } +} diff --git a/dotnet/samples/GettingStarted/AgentWithOpenAI/Agent_OpenAI_Step03_CreateFromChatClient/README.md b/dotnet/samples/GettingStarted/AgentWithOpenAI/Agent_OpenAI_Step03_CreateFromChatClient/README.md new file mode 100644 index 0000000000..a4d9dd76a5 --- /dev/null +++ b/dotnet/samples/GettingStarted/AgentWithOpenAI/Agent_OpenAI_Step03_CreateFromChatClient/README.md @@ -0,0 +1,22 @@ +# Creating an Agent from a ChatClient + +This sample demonstrates how to create an AI agent directly from an `OpenAI.Chat.ChatClient` instance using the `OpenAIChatClientAgent` class. + +## What This Sample Shows + +- **Direct ChatClient Creation**: Shows how to create an `OpenAI.Chat.ChatClient` from `OpenAI.OpenAIClient` and then use it to instantiate an agent +- **OpenAIChatClientAgent**: Demonstrates using the OpenAI SDK primitives instead of the ones from Microsoft.Extensions.AI and Microsoft.Agents.AI abstractions +- **Full Agent Capabilities**: Shows both regular and streaming invocation of the agent + +## Running the Sample + +1. Set the required environment variables: + ```bash + set OPENAI_API_KEY=your_api_key_here + set OPENAI_MODEL=gpt-4o-mini + ``` + +2. Run the sample: + ```bash + dotnet run + ``` diff --git a/dotnet/samples/GettingStarted/AgentWithOpenAI/Agent_OpenAI_Step04_CreateFromOpenAIResponseClient/Agent_OpenAI_Step04_CreateFromOpenAIResponseClient.csproj b/dotnet/samples/GettingStarted/AgentWithOpenAI/Agent_OpenAI_Step04_CreateFromOpenAIResponseClient/Agent_OpenAI_Step04_CreateFromOpenAIResponseClient.csproj new file mode 100644 index 0000000000..eeda3eef6f --- /dev/null +++ b/dotnet/samples/GettingStarted/AgentWithOpenAI/Agent_OpenAI_Step04_CreateFromOpenAIResponseClient/Agent_OpenAI_Step04_CreateFromOpenAIResponseClient.csproj @@ -0,0 +1,15 @@ + + + + Exe + net10.0 + + enable + enable + + + + + + + diff --git a/dotnet/src/Microsoft.Agents.AI.OpenAI/OpenAIResponseClientAgent.cs b/dotnet/samples/GettingStarted/AgentWithOpenAI/Agent_OpenAI_Step04_CreateFromOpenAIResponseClient/OpenAIResponseClientAgent.cs similarity index 95% rename from dotnet/src/Microsoft.Agents.AI.OpenAI/OpenAIResponseClientAgent.cs rename to dotnet/samples/GettingStarted/AgentWithOpenAI/Agent_OpenAI_Step04_CreateFromOpenAIResponseClient/OpenAIResponseClientAgent.cs index 8c5603fb05..456de02836 100644 --- a/dotnet/src/Microsoft.Agents.AI.OpenAI/OpenAIResponseClientAgent.cs +++ b/dotnet/samples/GettingStarted/AgentWithOpenAI/Agent_OpenAI_Step04_CreateFromOpenAIResponseClient/OpenAIResponseClientAgent.cs @@ -4,10 +4,9 @@ using Microsoft.Agents.AI; using Microsoft.Extensions.AI; using Microsoft.Extensions.Logging; -using Microsoft.Shared.Diagnostics; using OpenAI.Responses; -namespace OpenAI; +namespace OpenAIResponseClientSample; /// /// Provides an backed by an OpenAI Responses implementation. @@ -32,7 +31,7 @@ public OpenAIResponseClientAgent( { Name = name, Description = description, - Instructions = instructions, + ChatOptions = new ChatOptions() { Instructions = instructions }, }, loggerFactory) { } @@ -45,7 +44,7 @@ public OpenAIResponseClientAgent( /// Optional instance of public OpenAIResponseClientAgent( OpenAIResponseClient client, ChatClientAgentOptions options, ILoggerFactory? loggerFactory = null) : - base(new ChatClientAgent(Throw.IfNull(client).AsIChatClient(), options, loggerFactory)) + base(new ChatClientAgent((client ?? throw new ArgumentNullException(nameof(client))).AsIChatClient(), options, loggerFactory)) { } diff --git a/dotnet/samples/GettingStarted/AgentWithOpenAI/Agent_OpenAI_Step04_CreateFromOpenAIResponseClient/Program.cs b/dotnet/samples/GettingStarted/AgentWithOpenAI/Agent_OpenAI_Step04_CreateFromOpenAIResponseClient/Program.cs new file mode 100644 index 0000000000..89a96bc0fb --- /dev/null +++ b/dotnet/samples/GettingStarted/AgentWithOpenAI/Agent_OpenAI_Step04_CreateFromOpenAIResponseClient/Program.cs @@ -0,0 +1,32 @@ +// Copyright (c) Microsoft. All rights reserved. + +// This sample demonstrates how to create OpenAIResponseClientAgent directly from an OpenAIResponseClient instance. + +using OpenAI; +using OpenAI.Responses; +using OpenAIResponseClientSample; + +var apiKey = Environment.GetEnvironmentVariable("OPENAI_API_KEY") ?? throw new InvalidOperationException("OPENAI_API_KEY is not set."); +var model = Environment.GetEnvironmentVariable("OPENAI_MODEL") ?? "gpt-4o-mini"; + +// Create an OpenAIResponseClient directly from OpenAIClient +OpenAIResponseClient responseClient = new OpenAIClient(apiKey).GetOpenAIResponseClient(model); + +// Create an agent directly from the OpenAIResponseClient using OpenAIResponseClientAgent +OpenAIResponseClientAgent agent = new(responseClient, instructions: "You are good at telling jokes.", name: "Joker"); + +ResponseItem userMessage = ResponseItem.CreateUserMessageItem("Tell me a joke about a pirate."); + +// Invoke the agent and output the text result. +OpenAIResponse response = await agent.RunAsync([userMessage]); +Console.WriteLine(response.GetOutputText()); + +// Invoke the agent with streaming support. +IAsyncEnumerable responseUpdates = agent.RunStreamingAsync([userMessage]); +await foreach (StreamingResponseUpdate responseUpdate in responseUpdates) +{ + if (responseUpdate is StreamingResponseOutputTextDeltaUpdate textUpdate) + { + Console.WriteLine(textUpdate.Delta); + } +} diff --git a/dotnet/samples/GettingStarted/AgentWithOpenAI/Agent_OpenAI_Step04_CreateFromOpenAIResponseClient/README.md b/dotnet/samples/GettingStarted/AgentWithOpenAI/Agent_OpenAI_Step04_CreateFromOpenAIResponseClient/README.md new file mode 100644 index 0000000000..32e19caa62 --- /dev/null +++ b/dotnet/samples/GettingStarted/AgentWithOpenAI/Agent_OpenAI_Step04_CreateFromOpenAIResponseClient/README.md @@ -0,0 +1,22 @@ +# Creating an Agent from an OpenAIResponseClient + +This sample demonstrates how to create an AI agent directly from an `OpenAI.Responses.OpenAIResponseClient` instance using the `OpenAIResponseClientAgent` class. + +## What This Sample Shows + +- **Direct OpenAIResponseClient Creation**: Shows how to create an `OpenAI.Responses.OpenAIResponseClient` from `OpenAI.OpenAIClient` and then use it to instantiate an agent +- **OpenAIResponseClientAgent**: Demonstrates using the OpenAI SDK primitives instead of the ones from Microsoft.Extensions.AI and Microsoft.Agents.AI abstractions +- **Full Agent Capabilities**: Shows both regular and streaming invocation of the agent + +## Running the Sample + +1. Set the required environment variables: + ```bash + set OPENAI_API_KEY=your_api_key_here + set OPENAI_MODEL=gpt-4o-mini + ``` + +2. Run the sample: + ```bash + dotnet run + ``` diff --git a/dotnet/samples/GettingStarted/AgentWithOpenAI/Agent_OpenAI_Step05_Conversation/Agent_OpenAI_Step05_Conversation.csproj b/dotnet/samples/GettingStarted/AgentWithOpenAI/Agent_OpenAI_Step05_Conversation/Agent_OpenAI_Step05_Conversation.csproj new file mode 100644 index 0000000000..eeda3eef6f --- /dev/null +++ b/dotnet/samples/GettingStarted/AgentWithOpenAI/Agent_OpenAI_Step05_Conversation/Agent_OpenAI_Step05_Conversation.csproj @@ -0,0 +1,15 @@ + + + + Exe + net10.0 + + enable + enable + + + + + + + diff --git a/dotnet/samples/GettingStarted/AgentWithOpenAI/Agent_OpenAI_Step05_Conversation/Program.cs b/dotnet/samples/GettingStarted/AgentWithOpenAI/Agent_OpenAI_Step05_Conversation/Program.cs new file mode 100644 index 0000000000..9f81a27dda --- /dev/null +++ b/dotnet/samples/GettingStarted/AgentWithOpenAI/Agent_OpenAI_Step05_Conversation/Program.cs @@ -0,0 +1,98 @@ +// Copyright (c) Microsoft. All rights reserved. + +// This sample demonstrates how to maintain conversation state using the OpenAIResponseClientAgent +// and AgentThread. By passing the same thread to multiple agent invocations, the agent +// automatically maintains the conversation history, allowing the AI model to understand +// context from previous exchanges. + +using System.ClientModel; +using System.ClientModel.Primitives; +using System.Text.Json; +using Microsoft.Agents.AI; +using Microsoft.Extensions.AI; +using OpenAI; +using OpenAI.Chat; +using OpenAI.Conversations; + +string apiKey = Environment.GetEnvironmentVariable("OPENAI_API_KEY") ?? throw new InvalidOperationException("OPENAI_API_KEY is not set."); +string model = Environment.GetEnvironmentVariable("OPENAI_MODEL") ?? "gpt-4o-mini"; + +// Create a ConversationClient directly from OpenAIClient +OpenAIClient openAIClient = new(apiKey); +ConversationClient conversationClient = openAIClient.GetConversationClient(); + +// Create an agent directly from the OpenAIResponseClient using OpenAIResponseClientAgent +ChatClientAgent agent = new(openAIClient.GetOpenAIResponseClient(model).AsIChatClient(), instructions: "You are a helpful assistant.", name: "ConversationAgent"); + +ClientResult createConversationResult = await conversationClient.CreateConversationAsync(BinaryContent.Create(BinaryData.FromString("{}"))); + +using JsonDocument createConversationResultAsJson = JsonDocument.Parse(createConversationResult.GetRawResponse().Content.ToString()); +string conversationId = createConversationResultAsJson.RootElement.GetProperty("id"u8)!.GetString()!; + +// Create a thread for the conversation - this enables conversation state management for subsequent turns +AgentThread thread = agent.GetNewThread(conversationId); + +Console.WriteLine("=== Multi-turn Conversation Demo ===\n"); + +// First turn: Ask about a topic +Console.WriteLine("User: What is the capital of France?"); +UserChatMessage firstMessage = new("What is the capital of France?"); + +// After this call, the conversation state associated in the options is stored in 'thread' and used in subsequent calls +ChatCompletion firstResponse = await agent.RunAsync([firstMessage], thread); +Console.WriteLine($"Assistant: {firstResponse.Content.Last().Text}\n"); + +// Second turn: Follow-up question that relies on conversation context +Console.WriteLine("User: What famous landmarks are located there?"); +UserChatMessage secondMessage = new("What famous landmarks are located there?"); + +ChatCompletion secondResponse = await agent.RunAsync([secondMessage], thread); +Console.WriteLine($"Assistant: {secondResponse.Content.Last().Text}\n"); + +// Third turn: Another follow-up that demonstrates context continuity +Console.WriteLine("User: How tall is the most famous one?"); +UserChatMessage thirdMessage = new("How tall is the most famous one?"); + +ChatCompletion thirdResponse = await agent.RunAsync([thirdMessage], thread); +Console.WriteLine($"Assistant: {thirdResponse.Content.Last().Text}\n"); + +Console.WriteLine("=== End of Conversation ==="); + +// Show full conversation history +Console.WriteLine("Full Conversation History:"); +ClientResult getConversationResult = await conversationClient.GetConversationAsync(conversationId); + +Console.WriteLine("Conversation created."); +Console.WriteLine($" Conversation ID: {conversationId}"); +Console.WriteLine(); + +CollectionResult getConversationItemsResults = conversationClient.GetConversationItems(conversationId); +foreach (ClientResult result in getConversationItemsResults.GetRawPages()) +{ + Console.WriteLine("Message contents retrieved. Order is most recent first by default."); + using JsonDocument getConversationItemsResultAsJson = JsonDocument.Parse(result.GetRawResponse().Content.ToString()); + foreach (JsonElement element in getConversationItemsResultAsJson.RootElement.GetProperty("data").EnumerateArray()) + { + string messageId = element.GetProperty("id"u8).ToString(); + string messageRole = element.GetProperty("role"u8).ToString(); + Console.WriteLine($" Message ID: {messageId}"); + Console.WriteLine($" Message Role: {messageRole}"); + + foreach (var content in element.GetProperty("content").EnumerateArray()) + { + string messageContentText = content.GetProperty("text"u8).ToString(); + Console.WriteLine($" Message Text: {messageContentText}"); + } + Console.WriteLine(); + } +} + +ClientResult deleteConversationResult = conversationClient.DeleteConversation(conversationId); +using JsonDocument deleteConversationResultAsJson = JsonDocument.Parse(deleteConversationResult.GetRawResponse().Content.ToString()); +bool deleted = deleteConversationResultAsJson.RootElement + .GetProperty("deleted"u8) + .GetBoolean(); + +Console.WriteLine("Conversation deleted."); +Console.WriteLine($" Deleted: {deleted}"); +Console.WriteLine(); diff --git a/dotnet/samples/GettingStarted/AgentWithOpenAI/Agent_OpenAI_Step05_Conversation/README.md b/dotnet/samples/GettingStarted/AgentWithOpenAI/Agent_OpenAI_Step05_Conversation/README.md new file mode 100644 index 0000000000..c279ba2c17 --- /dev/null +++ b/dotnet/samples/GettingStarted/AgentWithOpenAI/Agent_OpenAI_Step05_Conversation/README.md @@ -0,0 +1,90 @@ +# Managing Conversation State with OpenAI + +This sample demonstrates how to maintain conversation state across multiple turns using the Agent Framework with OpenAI's Conversation API. + +## What This Sample Shows + +- **Conversation State Management**: Shows how to use `ConversationClient` and `AgentThread` to maintain conversation context across multiple agent invocations +- **Multi-turn Conversations**: Demonstrates follow-up questions that rely on context from previous messages in the conversation +- **Server-Side Storage**: Uses OpenAI's Conversation API to manage conversation history server-side, allowing the model to access previous messages without resending them +- **Conversation Lifecycle**: Demonstrates creating, retrieving, and deleting conversations + +## Key Concepts + +### ConversationClient for Server-Side Storage + +The `ConversationClient` manages conversations on OpenAI's servers: + +```csharp +// Create a ConversationClient from OpenAIClient +OpenAIClient openAIClient = new(apiKey); +ConversationClient conversationClient = openAIClient.GetConversationClient(); + +// Create a new conversation +ClientResult createConversationResult = await conversationClient.CreateConversationAsync(BinaryContent.Create(BinaryData.FromString("{}"))); +``` + +### AgentThread for Conversation State + +The `AgentThread` works with `ChatClientAgentRunOptions` to link the agent to a server-side conversation: + +```csharp +// Set up agent run options with the conversation ID +ChatClientAgentRunOptions agentRunOptions = new() { ChatOptions = new ChatOptions() { ConversationId = conversationId } }; + +// Create a thread for the conversation +AgentThread thread = agent.GetNewThread(); + +// First call links the thread to the conversation +ChatCompletion firstResponse = await agent.RunAsync([firstMessage], thread, agentRunOptions); + +// Subsequent calls use the thread without needing to pass options again +ChatCompletion secondResponse = await agent.RunAsync([secondMessage], thread); +``` + +### Retrieving Conversation History + +You can retrieve the full conversation history from the server: + +```csharp +CollectionResult getConversationItemsResults = conversationClient.GetConversationItems(conversationId); +foreach (ClientResult result in getConversationItemsResults.GetRawPages()) +{ + // Process conversation items +} +``` + +### How It Works + +1. **Create an OpenAI Client**: Initialize an `OpenAIClient` with your API key +2. **Create a Conversation**: Use `ConversationClient` to create a server-side conversation +3. **Create an Agent**: Initialize an `OpenAIResponseClientAgent` with the desired model and instructions +4. **Create a Thread**: Call `agent.GetNewThread()` to create a new conversation thread +5. **Link Thread to Conversation**: Pass `ChatClientAgentRunOptions` with the `ConversationId` on the first call +6. **Send Messages**: Subsequent calls to `agent.RunAsync()` only need the thread - context is maintained +7. **Cleanup**: Delete the conversation when done using `conversationClient.DeleteConversation()` + +## Running the Sample + +1. Set the required environment variables: + ```powershell + $env:OPENAI_API_KEY = "your_api_key_here" + $env:OPENAI_MODEL = "gpt-4o-mini" + ``` + +2. Run the sample: + ```powershell + dotnet run + ``` + +## Expected Output + +The sample demonstrates a three-turn conversation where each follow-up question relies on context from previous messages: + +1. First question asks about the capital of France +2. Second question asks about landmarks "there" - requiring understanding of the previous answer +3. Third question asks about "the most famous one" - requiring context from both previous turns + +After the conversation, the sample retrieves and displays the full conversation history from the server, then cleans up by deleting the conversation. + +This demonstrates that the conversation state is properly maintained across multiple agent invocations using OpenAI's server-side conversation storage. diff --git a/dotnet/samples/GettingStarted/AgentWithOpenAI/README.md b/dotnet/samples/GettingStarted/AgentWithOpenAI/README.md index 4ed609ae81..019af7f2b6 100644 --- a/dotnet/samples/GettingStarted/AgentWithOpenAI/README.md +++ b/dotnet/samples/GettingStarted/AgentWithOpenAI/README.md @@ -10,5 +10,8 @@ Agent Framework provides additional support to allow OpenAI developers to use th |Sample|Description| |---|---| -|[Creating an AIAgent](./Agent_OpenAI_Step01_Running/)|This sample demonstrates how to create and run a basic agent instructions with native OpenAI SDK types.| - +|[Creating an AIAgent](./Agent_OpenAI_Step01_Running/)|This sample demonstrates how to create and run a basic agent with native OpenAI SDK types. Shows both regular and streaming invocation of the agent.| +|[Using Reasoning Capabilities](./Agent_OpenAI_Step02_Reasoning/)|This sample demonstrates how to create an AI agent with reasoning capabilities using OpenAI's reasoning models and response types.| +|[Creating an Agent from a ChatClient](./Agent_OpenAI_Step03_CreateFromChatClient/)|This sample demonstrates how to create an AI agent directly from an OpenAI.Chat.ChatClient instance using OpenAIChatClientAgent.| +|[Creating an Agent from an OpenAIResponseClient](./Agent_OpenAI_Step04_CreateFromOpenAIResponseClient/)|This sample demonstrates how to create an AI agent directly from an OpenAI.Responses.OpenAIResponseClient instance using OpenAIResponseClientAgent.| +|[Managing Conversation State](./Agent_OpenAI_Step05_Conversation/)|This sample demonstrates how to maintain conversation state across multiple turns using the AgentThread for context continuity.| \ No newline at end of file diff --git a/dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step01_BasicTextRAG/AgentWithRAG_Step01_BasicTextRAG.csproj b/dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step01_BasicTextRAG/AgentWithRAG_Step01_BasicTextRAG.csproj index 0c8a9f2dfc..860089b621 100644 --- a/dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step01_BasicTextRAG/AgentWithRAG_Step01_BasicTextRAG.csproj +++ b/dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step01_BasicTextRAG/AgentWithRAG_Step01_BasicTextRAG.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable diff --git a/dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step01_BasicTextRAG/Program.cs b/dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step01_BasicTextRAG/Program.cs index ec665325a7..42015d87cd 100644 --- a/dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step01_BasicTextRAG/Program.cs +++ b/dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step01_BasicTextRAG/Program.cs @@ -8,12 +8,11 @@ using Azure.AI.OpenAI; using Azure.Identity; using Microsoft.Agents.AI; -using Microsoft.Agents.AI.Data; using Microsoft.Agents.AI.Samples; using Microsoft.Extensions.AI; using Microsoft.Extensions.VectorData; using Microsoft.SemanticKernel.Connectors.InMemory; -using OpenAI; +using OpenAI.Chat; var endpoint = Environment.GetEnvironmentVariable("AZURE_OPENAI_ENDPOINT") ?? throw new InvalidOperationException("AZURE_OPENAI_ENDPOINT is not set."); var deploymentName = Environment.GetEnvironmentVariable("AZURE_OPENAI_DEPLOYMENT_NAME") ?? "gpt-4o-mini"; @@ -62,7 +61,7 @@ .GetChatClient(deploymentName) .CreateAIAgent(new ChatClientAgentOptions { - Instructions = "You are a helpful support specialist for Contoso Outdoors. Answer questions using the provided context and cite the source document when available.", + ChatOptions = new() { Instructions = "You are a helpful support specialist for Contoso Outdoors. Answer questions using the provided context and cite the source document when available." }, AIContextProviderFactory = ctx => new TextSearchProvider(SearchAdapter, ctx.SerializedState, ctx.JsonSerializerOptions, textSearchOptions) }); diff --git a/dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step01_BasicTextRAG/TextSearchStore/TextSearchStore.cs b/dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step01_BasicTextRAG/TextSearchStore/TextSearchStore.cs index 502c17dba1..82559ecf83 100644 --- a/dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step01_BasicTextRAG/TextSearchStore/TextSearchStore.cs +++ b/dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step01_BasicTextRAG/TextSearchStore/TextSearchStore.cs @@ -98,8 +98,8 @@ public TextSearchStore( // Create a definition so that we can use the dimensions provided at runtime. VectorStoreCollectionDefinition ragDocumentDefinition = new() { - Properties = new List() - { + Properties = + [ new VectorStoreKeyProperty("Key", this._options.KeyType ?? typeof(string)), new VectorStoreDataProperty("Namespaces", typeof(List)) { IsIndexed = true }, new VectorStoreDataProperty("SourceId", typeof(string)) { IsIndexed = true }, @@ -107,7 +107,7 @@ public TextSearchStore( new VectorStoreDataProperty("SourceName", typeof(string)), new VectorStoreDataProperty("SourceLink", typeof(string)), new VectorStoreVectorProperty("TextEmbedding", typeof(string), vectorDimensions), - } + ] }; this._vectorStoreRecordCollection = this._vectorStore.GetDynamicCollection(collectionName, ragDocumentDefinition); @@ -267,7 +267,7 @@ public async Task> SearchAsync(string query, int cancellationToken: cancellationToken); // Retrieve the documents from the search results. - List> searchResponseDocs = new(); + List> searchResponseDocs = []; await foreach (var searchResponseDoc in searchResult.WithCancellation(cancellationToken).ConfigureAwait(false)) { searchResponseDocs.Add(searchResponseDoc.Record); @@ -291,12 +291,8 @@ public async Task> SearchAsync(string query, int } // Retrieve the source text for the documents that need it. - var retrievalResponses = await this._options.SourceRetrievalCallback(sourceIdsToRetrieve).ConfigureAwait(false); - - if (retrievalResponses is null) - { + var retrievalResponses = await this._options.SourceRetrievalCallback(sourceIdsToRetrieve).ConfigureAwait(false) ?? throw new InvalidOperationException($"The {nameof(TextSearchStoreOptions.SourceRetrievalCallback)} must return a non-null value."); - } // Update the retrieved documents with the retrieved text. return searchResponseDocs.GroupJoin( diff --git a/dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step01_BasicTextRAG/TextSearchStore/TextSearchStoreOptions.cs b/dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step01_BasicTextRAG/TextSearchStore/TextSearchStoreOptions.cs index 53da092c82..d9b8761be6 100644 --- a/dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step01_BasicTextRAG/TextSearchStore/TextSearchStoreOptions.cs +++ b/dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step01_BasicTextRAG/TextSearchStore/TextSearchStoreOptions.cs @@ -107,15 +107,8 @@ public sealed class SourceRetrievalResponse /// The source text that was retrieved. public SourceRetrievalResponse(SourceRetrievalRequest request, string text) { - if (request == null) - { - throw new ArgumentNullException(nameof(request)); - } - - if (text == null) - { - throw new ArgumentNullException(nameof(text)); - } + ArgumentNullException.ThrowIfNull(request); + ArgumentNullException.ThrowIfNull(text); this.SourceId = request.SourceId; this.SourceLink = request.SourceLink; diff --git a/dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step02_ExternalDataSourceRAG/AgentWithRAG_Step02_ExternalDataSourceRAG.csproj b/dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step02_CustomVectorStoreRAG/AgentWithRAG_Step02_CustomVectorStoreRAG.csproj similarity index 92% rename from dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step02_ExternalDataSourceRAG/AgentWithRAG_Step02_ExternalDataSourceRAG.csproj rename to dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step02_CustomVectorStoreRAG/AgentWithRAG_Step02_CustomVectorStoreRAG.csproj index 56e2ad232b..33029395dd 100644 --- a/dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step02_ExternalDataSourceRAG/AgentWithRAG_Step02_ExternalDataSourceRAG.csproj +++ b/dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step02_CustomVectorStoreRAG/AgentWithRAG_Step02_CustomVectorStoreRAG.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable diff --git a/dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step02_ExternalDataSourceRAG/Program.cs b/dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step02_CustomVectorStoreRAG/Program.cs similarity index 94% rename from dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step02_ExternalDataSourceRAG/Program.cs rename to dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step02_CustomVectorStoreRAG/Program.cs index 4e8fbf0bde..f20e42f01d 100644 --- a/dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step02_ExternalDataSourceRAG/Program.cs +++ b/dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step02_CustomVectorStoreRAG/Program.cs @@ -1,17 +1,16 @@ // Copyright (c) Microsoft. All rights reserved. -// This sample shows how to use Qdrant to add retrieval augmented generation (RAG) capabilities to an AI agent. +// This sample shows how to use Qdrant with a custom schema to add retrieval augmented generation (RAG) capabilities to an AI agent. // While the sample is using Qdrant, it can easily be replaced with any other vector store that implements the Microsoft.Extensions.VectorData abstractions. // The TextSearchProvider runs a search against the vector store before each model invocation and injects the results into the model context. using Azure.AI.OpenAI; using Azure.Identity; using Microsoft.Agents.AI; -using Microsoft.Agents.AI.Data; using Microsoft.Extensions.AI; using Microsoft.Extensions.VectorData; using Microsoft.SemanticKernel.Connectors.Qdrant; -using OpenAI; +using OpenAI.Chat; using Qdrant.Client; var endpoint = Environment.GetEnvironmentVariable("AZURE_OPENAI_ENDPOINT") ?? throw new InvalidOperationException("AZURE_OPENAI_ENDPOINT is not set."); @@ -71,7 +70,7 @@ .GetChatClient(deploymentName) .CreateAIAgent(new ChatClientAgentOptions { - Instructions = "You are a helpful support specialist for the Microsoft Agent Framework. Answer questions using the provided context and cite the source document when available. Keep responses brief.", + ChatOptions = new() { Instructions = "You are a helpful support specialist for the Microsoft Agent Framework. Answer questions using the provided context and cite the source document when available. Keep responses brief." }, AIContextProviderFactory = ctx => new TextSearchProvider(SearchAdapter, ctx.SerializedState, ctx.JsonSerializerOptions, textSearchOptions) }); diff --git a/dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step02_ExternalDataSourceRAG/README.md b/dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step02_CustomVectorStoreRAG/README.md similarity index 99% rename from dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step02_ExternalDataSourceRAG/README.md rename to dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step02_CustomVectorStoreRAG/README.md index 1817f0d8ca..131adde82b 100644 --- a/dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step02_ExternalDataSourceRAG/README.md +++ b/dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step02_CustomVectorStoreRAG/README.md @@ -6,7 +6,7 @@ This sample uses Qdrant for the vector store, but this can easily be swapped out ## Prerequisites -- .NET 8.0 SDK or later +- .NET 10 SDK or later - Azure OpenAI service endpoint - Both a chat completion and embedding deployment configured in the Azure OpenAI resource - Azure CLI installed and authenticated (for Azure credential authentication) diff --git a/dotnet/samples/GettingStarted/Agents/Agent_Step03.1_UsingFunctionTools/Agent_Step03.1_UsingFunctionTools.csproj b/dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step03_CustomRAGDataSource/AgentWithRAG_Step03_CustomRAGDataSource.csproj similarity index 91% rename from dotnet/samples/GettingStarted/Agents/Agent_Step03.1_UsingFunctionTools/Agent_Step03.1_UsingFunctionTools.csproj rename to dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step03_CustomRAGDataSource/AgentWithRAG_Step03_CustomRAGDataSource.csproj index 8298cfe6e8..0f9de7c359 100644 --- a/dotnet/samples/GettingStarted/Agents/Agent_Step03.1_UsingFunctionTools/Agent_Step03.1_UsingFunctionTools.csproj +++ b/dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step03_CustomRAGDataSource/AgentWithRAG_Step03_CustomRAGDataSource.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable diff --git a/dotnet/samples/Catalog/AgentWithTextSearchRag/Program.cs b/dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step03_CustomRAGDataSource/Program.cs similarity index 87% rename from dotnet/samples/Catalog/AgentWithTextSearchRag/Program.cs rename to dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step03_CustomRAGDataSource/Program.cs index 65f3a9e98f..e9a62e382f 100644 --- a/dotnet/samples/Catalog/AgentWithTextSearchRag/Program.cs +++ b/dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step03_CustomRAGDataSource/Program.cs @@ -1,15 +1,16 @@ // Copyright (c) Microsoft. All rights reserved. // This sample shows how to use TextSearchProvider to add retrieval augmented generation (RAG) -// capabilities to an AI agent. The provider runs a search against an external knowledge base +// capabilities to an AI agent. This shows a mock implementation of a search function, +// which can be replaced with any custom search logic to query any external knowledge base. +// The provider invokes the custom search function // before each model invocation and injects the results into the model context. using Azure.AI.OpenAI; using Azure.Identity; using Microsoft.Agents.AI; -using Microsoft.Agents.AI.Data; using Microsoft.Extensions.AI; -using OpenAI; +using OpenAI.Chat; var endpoint = Environment.GetEnvironmentVariable("AZURE_OPENAI_ENDPOINT") ?? throw new InvalidOperationException("AZURE_OPENAI_ENDPOINT is not set."); var deploymentName = Environment.GetEnvironmentVariable("AZURE_OPENAI_DEPLOYMENT_NAME") ?? "gpt-4o-mini"; @@ -27,7 +28,7 @@ .GetChatClient(deploymentName) .CreateAIAgent(new ChatClientAgentOptions { - Instructions = "You are a helpful support specialist for Contoso Outdoors. Answer questions using the provided context and cite the source document when available.", + ChatOptions = new() { Instructions = "You are a helpful support specialist for Contoso Outdoors. Answer questions using the provided context and cite the source document when available." }, AIContextProviderFactory = ctx => new TextSearchProvider(MockSearchAsync, ctx.SerializedState, ctx.JsonSerializerOptions, textSearchOptions) }); @@ -46,7 +47,7 @@ { // The mock search inspects the user's question and returns pre-defined snippets // that resemble documents stored in an external knowledge source. - List results = new(); + List results = []; if (query.Contains("return", StringComparison.OrdinalIgnoreCase) || query.Contains("refund", StringComparison.OrdinalIgnoreCase)) { diff --git a/dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step04_FoundryServiceRAG/AgentWithRAG_Step04_FoundryServiceRAG.csproj b/dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step04_FoundryServiceRAG/AgentWithRAG_Step04_FoundryServiceRAG.csproj new file mode 100644 index 0000000000..d90e1c394b --- /dev/null +++ b/dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step04_FoundryServiceRAG/AgentWithRAG_Step04_FoundryServiceRAG.csproj @@ -0,0 +1,26 @@ + + + + Exe + net10.0 + + enable + enable + + + + + + + + + + + + + + Always + + + + diff --git a/dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step04_FoundryServiceRAG/Program.cs b/dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step04_FoundryServiceRAG/Program.cs new file mode 100644 index 0000000000..0989394185 --- /dev/null +++ b/dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step04_FoundryServiceRAG/Program.cs @@ -0,0 +1,60 @@ +// Copyright (c) Microsoft. All rights reserved. + +// This sample shows how to use the built in RAG capabilities that the Foundry service provides when using AI Agents provided by Foundry. + +using System.ClientModel; +using Azure.AI.Projects; +using Azure.Identity; +using Microsoft.Agents.AI; +using Microsoft.Extensions.AI; +using OpenAI; +using OpenAI.Files; +using OpenAI.VectorStores; + +var endpoint = Environment.GetEnvironmentVariable("AZURE_FOUNDRY_PROJECT_ENDPOINT") ?? throw new InvalidOperationException("AZURE_FOUNDRY_PROJECT_ENDPOINT is not set."); +var deploymentName = Environment.GetEnvironmentVariable("AZURE_FOUNDRY_PROJECT_DEPLOYMENT_NAME") ?? "gpt-4o-mini"; + +// Create an AI Project client and get an OpenAI client that works with the foundry service. +AIProjectClient aiProjectClient = new( + new Uri(endpoint), + new AzureCliCredential()); +OpenAIClient openAIClient = aiProjectClient.GetProjectOpenAIClient(); + +// Upload the file that contains the data to be used for RAG to the Foundry service. +OpenAIFileClient fileClient = openAIClient.GetOpenAIFileClient(); +ClientResult uploadResult = await fileClient.UploadFileAsync( + filePath: "contoso-outdoors-knowledge-base.md", + purpose: FileUploadPurpose.Assistants); + +// Create a vector store in the Foundry service using the uploaded file. +VectorStoreClient vectorStoreClient = openAIClient.GetVectorStoreClient(); +ClientResult vectorStoreCreate = await vectorStoreClient.CreateVectorStoreAsync(options: new VectorStoreCreationOptions() +{ + Name = "contoso-outdoors-knowledge-base", + FileIds = { uploadResult.Value.Id } +}); + +var fileSearchTool = new HostedFileSearchTool() { Inputs = [new HostedVectorStoreContent(vectorStoreCreate.Value.Id)] }; + +AIAgent agent = await aiProjectClient + .CreateAIAgentAsync( + model: deploymentName, + name: "AskContoso", + instructions: "You are a helpful support specialist for Contoso Outdoors. Answer questions using the provided context and cite the source document when available.", + tools: [fileSearchTool]); + +AgentThread thread = agent.GetNewThread(); + +Console.WriteLine(">> Asking about returns\n"); +Console.WriteLine(await agent.RunAsync("Hi! I need help understanding the return policy.", thread)); + +Console.WriteLine("\n>> Asking about shipping\n"); +Console.WriteLine(await agent.RunAsync("How long does standard shipping usually take?", thread)); + +Console.WriteLine("\n>> Asking about product care\n"); +Console.WriteLine(await agent.RunAsync("What is the best way to maintain the TrailRunner tent fabric?", thread)); + +// Cleanup +await fileClient.DeleteFileAsync(uploadResult.Value.Id); +await vectorStoreClient.DeleteVectorStoreAsync(vectorStoreCreate.Value.Id); +await aiProjectClient.Agents.DeleteAgentAsync(agent.Name); diff --git a/dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step04_FoundryServiceRAG/contoso-outdoors-knowledge-base.md b/dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step04_FoundryServiceRAG/contoso-outdoors-knowledge-base.md new file mode 100644 index 0000000000..901e45b4dd --- /dev/null +++ b/dotnet/samples/GettingStarted/AgentWithRAG/AgentWithRAG_Step04_FoundryServiceRAG/contoso-outdoors-knowledge-base.md @@ -0,0 +1,19 @@ +# Contoso Outdoors Knowledge Base + +## Contoso Outdoors Return Policy + +Customers may return any item within 30 days of delivery. Items should be unused and include original packaging. Refunds are issued to the original payment method within 5 business days of inspection. + +## Contoso Outdoors Shipping Guide + +Standard shipping is free on orders over $50 and typically arrives in 3-5 business days within the continental United States. Expedited options are available at checkout. + +## Product Information + +### TrailRunner Tent + +The TrailRunner Tent is a lightweight, 2-person tent designed for easy setup and durability. It features waterproof materials, ventilation windows, and a compact carry bag. + +#### Care Instructions + +Clean the tent fabric with lukewarm water and a non-detergent soap. Allow it to air dry completely before storage and avoid prolonged UV exposure to extend the lifespan of the waterproof coating. \ No newline at end of file diff --git a/dotnet/samples/GettingStarted/AgentWithRAG/README.md b/dotnet/samples/GettingStarted/AgentWithRAG/README.md index f45c2c2540..d606ac767c 100644 --- a/dotnet/samples/GettingStarted/AgentWithRAG/README.md +++ b/dotnet/samples/GettingStarted/AgentWithRAG/README.md @@ -5,4 +5,6 @@ These samples show how to create an agent with the Agent Framework that uses Ret |Sample|Description| |---|---| |[Basic Text RAG](./AgentWithRAG_Step01_BasicTextRAG/)|This sample demonstrates how to create and run a basic agent with simple text Retrieval Augmented Generation (RAG).| -|[RAG with external Vector Store and custom schema](./AgentWithRAG_Step02_ExternalDataSourceRAG/)|This sample demonstrates how to create and run an agent that uses Retrieval Augmented Generation (RAG) with an external vector store. It also uses a custom schema for the documents stored in the vector store.| +|[RAG with Vector Store and custom schema](./AgentWithRAG_Step02_CustomVectorStoreRAG/)|This sample demonstrates how to create and run an agent that uses Retrieval Augmented Generation (RAG) with a vector store. It also uses a custom schema for the documents stored in the vector store.| +|[RAG with custom RAG data source](./AgentWithRAG_Step03_CustomRAGDataSource/)|This sample demonstrates how to create and run an agent that uses Retrieval Augmented Generation (RAG) with a custom RAG data source.| +|[RAG with Foundry VectorStore service](./AgentWithRAG_Step04_FoundryServiceRAG/)|This sample demonstrates how to create and run an agent that uses Retrieval Augmented Generation (RAG) with the Foundry VectorStore service.| diff --git a/dotnet/samples/GettingStarted/Agents/Agent_Step01_Running/Agent_Step01_Running.csproj b/dotnet/samples/GettingStarted/Agents/Agent_Step01_Running/Agent_Step01_Running.csproj index 8298cfe6e8..0f9de7c359 100644 --- a/dotnet/samples/GettingStarted/Agents/Agent_Step01_Running/Agent_Step01_Running.csproj +++ b/dotnet/samples/GettingStarted/Agents/Agent_Step01_Running/Agent_Step01_Running.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable diff --git a/dotnet/samples/GettingStarted/Agents/Agent_Step01_Running/Program.cs b/dotnet/samples/GettingStarted/Agents/Agent_Step01_Running/Program.cs index c67756299c..889045c228 100644 --- a/dotnet/samples/GettingStarted/Agents/Agent_Step01_Running/Program.cs +++ b/dotnet/samples/GettingStarted/Agents/Agent_Step01_Running/Program.cs @@ -5,7 +5,7 @@ using Azure.AI.OpenAI; using Azure.Identity; using Microsoft.Agents.AI; -using OpenAI; +using OpenAI.Chat; var endpoint = Environment.GetEnvironmentVariable("AZURE_OPENAI_ENDPOINT") ?? throw new InvalidOperationException("AZURE_OPENAI_ENDPOINT is not set."); var deploymentName = Environment.GetEnvironmentVariable("AZURE_OPENAI_DEPLOYMENT_NAME") ?? "gpt-4o-mini"; diff --git a/dotnet/samples/GettingStarted/Agents/Agent_Step02_MultiturnConversation/Agent_Step02_MultiturnConversation.csproj b/dotnet/samples/GettingStarted/Agents/Agent_Step02_MultiturnConversation/Agent_Step02_MultiturnConversation.csproj index 8298cfe6e8..0f9de7c359 100644 --- a/dotnet/samples/GettingStarted/Agents/Agent_Step02_MultiturnConversation/Agent_Step02_MultiturnConversation.csproj +++ b/dotnet/samples/GettingStarted/Agents/Agent_Step02_MultiturnConversation/Agent_Step02_MultiturnConversation.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable diff --git a/dotnet/samples/GettingStarted/Agents/Agent_Step02_MultiturnConversation/Program.cs b/dotnet/samples/GettingStarted/Agents/Agent_Step02_MultiturnConversation/Program.cs index 626a3e98c4..e27a5bb36d 100644 --- a/dotnet/samples/GettingStarted/Agents/Agent_Step02_MultiturnConversation/Program.cs +++ b/dotnet/samples/GettingStarted/Agents/Agent_Step02_MultiturnConversation/Program.cs @@ -5,7 +5,7 @@ using Azure.AI.OpenAI; using Azure.Identity; using Microsoft.Agents.AI; -using OpenAI; +using OpenAI.Chat; var endpoint = Environment.GetEnvironmentVariable("AZURE_OPENAI_ENDPOINT") ?? throw new InvalidOperationException("AZURE_OPENAI_ENDPOINT is not set."); var deploymentName = Environment.GetEnvironmentVariable("AZURE_OPENAI_DEPLOYMENT_NAME") ?? "gpt-4o-mini"; diff --git a/dotnet/samples/GettingStarted/Agents/Agent_Step03.2_UsingFunctionTools_FromOpenAPI/OpenAPISpec.json b/dotnet/samples/GettingStarted/Agents/Agent_Step03.2_UsingFunctionTools_FromOpenAPI/OpenAPISpec.json deleted file mode 100644 index 84715914da..0000000000 --- a/dotnet/samples/GettingStarted/Agents/Agent_Step03.2_UsingFunctionTools_FromOpenAPI/OpenAPISpec.json +++ /dev/null @@ -1,354 +0,0 @@ -{ - "openapi": "3.0.1", - "info": { - "title": "Github Versions API", - "version": "1.0.0" - }, - "servers": [ - { - "url": "https://api.github.com" - } - ], - "components": { - "schemas": { - "basic-error": { - "title": "Basic Error", - "description": "Basic Error", - "type": "object", - "properties": { - "message": { - "type": "string" - }, - "documentation_url": { - "type": "string" - }, - "url": { - "type": "string" - }, - "status": { - "type": "string" - } - } - }, - "label": { - "title": "Label", - "description": "Color-coded labels help you categorize and filter your issues (just like labels in Gmail).", - "type": "object", - "properties": { - "id": { - "description": "Unique identifier for the label.", - "type": "integer", - "format": "int64", - "example": 208045946 - }, - "node_id": { - "type": "string", - "example": "MDU6TGFiZWwyMDgwNDU5NDY=" - }, - "url": { - "description": "URL for the label", - "example": "https://api.github.com/repositories/42/labels/bug", - "type": "string", - "format": "uri" - }, - "name": { - "description": "The name of the label.", - "example": "bug", - "type": "string" - }, - "description": { - "description": "Optional description of the label, such as its purpose.", - "type": "string", - "example": "Something isn't working", - "nullable": true - }, - "color": { - "description": "6-character hex code, without the leading #, identifying the color", - "example": "FFFFFF", - "type": "string" - }, - "default": { - "description": "Whether this label comes by default in a new repository.", - "type": "boolean", - "example": true - } - }, - "required": [ - "id", - "node_id", - "url", - "name", - "description", - "color", - "default" - ] - }, - "tag": { - "title": "Tag", - "description": "Tag", - "type": "object", - "properties": { - "name": { - "type": "string", - "example": "v0.1" - }, - "commit": { - "type": "object", - "properties": { - "sha": { - "type": "string" - }, - "url": { - "type": "string", - "format": "uri" - } - }, - "required": [ - "sha", - "url" - ] - }, - "zipball_url": { - "type": "string", - "format": "uri", - "example": "https://github.com/octocat/Hello-World/zipball/v0.1" - }, - "tarball_url": { - "type": "string", - "format": "uri", - "example": "https://github.com/octocat/Hello-World/tarball/v0.1" - }, - "node_id": { - "type": "string" - } - }, - "required": [ - "name", - "node_id", - "commit", - "zipball_url", - "tarball_url" - ] - } - }, - "examples": { - "label-items": { - "value": [ - { - "id": 208045946, - "node_id": "MDU6TGFiZWwyMDgwNDU5NDY=", - "url": "https://api.github.com/repos/octocat/Hello-World/labels/bug", - "name": "bug", - "description": "Something isn't working", - "color": "f29513", - "default": true - }, - { - "id": 208045947, - "node_id": "MDU6TGFiZWwyMDgwNDU5NDc=", - "url": "https://api.github.com/repos/octocat/Hello-World/labels/enhancement", - "name": "enhancement", - "description": "New feature or request", - "color": "a2eeef", - "default": false - } - ] - }, - "tag-items": { - "value": [ - { - "name": "v0.1", - "commit": { - "sha": "c5b97d5ae6c19d5c5df71a34c7fbeeda2479ccbc", - "url": "https://api.github.com/repos/octocat/Hello-World/commits/c5b97d5ae6c19d5c5df71a34c7fbeeda2479ccbc" - }, - "zipball_url": "https://github.com/octocat/Hello-World/zipball/v0.1", - "tarball_url": "https://github.com/octocat/Hello-World/tarball/v0.1", - "node_id": "MDQ6VXNlcjE=" - } - ] - } - }, - "parameters": { - "owner": { - "name": "owner", - "description": "The account owner of the repository. The name is not case sensitive.", - "in": "path", - "required": true, - "schema": { - "type": "string" - } - }, - "repo": { - "name": "repo", - "description": "The name of the repository without the `.git` extension. The name is not case sensitive.", - "in": "path", - "required": true, - "schema": { - "type": "string" - } - }, - "per-page": { - "name": "per_page", - "description": "The number of results per page (max 100). For more information, see \"[Using pagination in the REST API](https://docs.github.com/rest/using-the-rest-api/using-pagination-in-the-rest-api).\"", - "in": "query", - "schema": { - "type": "integer", - "default": 30 - } - }, - "page": { - "name": "page", - "description": "The page number of the results to fetch. For more information, see \"[Using pagination in the REST API](https://docs.github.com/rest/using-the-rest-api/using-pagination-in-the-rest-api).\"", - "in": "query", - "schema": { - "type": "integer", - "default": 1 - } - } - }, - "responses": { - "not_found": { - "description": "Resource not found", - "content": { - "application/json": { - "schema": { - "$ref": "#/components/schemas/basic-error" - } - } - } - } - }, - "headers": { - "link": { - "example": "; rel=\"next\", ; rel=\"last\"", - "schema": { - "type": "string" - } - } - } - }, - "paths": { - "/repos/{owner}/{repo}/tags": { - "get": { - "summary": "List repository tags", - "description": "", - "tags": [ - "repos" - ], - "operationId": "repos/list-tags", - "externalDocs": { - "description": "API method documentation", - "url": "https://docs.github.com/rest/repos/repos#list-repository-tags" - }, - "parameters": [ - { - "$ref": "#/components/parameters/owner" - }, - { - "$ref": "#/components/parameters/repo" - }, - { - "$ref": "#/components/parameters/per-page" - }, - { - "$ref": "#/components/parameters/page" - } - ], - "responses": { - "200": { - "description": "Response", - "content": { - "application/json": { - "schema": { - "type": "array", - "items": { - "$ref": "#/components/schemas/tag" - } - }, - "examples": { - "default": { - "$ref": "#/components/examples/tag-items" - } - } - } - }, - "headers": { - "Link": { - "$ref": "#/components/headers/link" - } - } - } - }, - "x-github": { - "githubCloudOnly": false, - "enabledForGitHubApps": true, - "category": "repos", - "subcategory": "repos" - } - } - }, - "/repos/{owner}/{repo}/labels": { - "get": { - "summary": "List labels for a repository", - "description": "Lists all labels for a repository.", - "tags": [ - "issues" - ], - "operationId": "issues/list-labels-for-repo", - "externalDocs": { - "description": "API method documentation", - "url": "https://docs.github.com/rest/issues/labels#list-labels-for-a-repository" - }, - "parameters": [ - { - "$ref": "#/components/parameters/owner" - }, - { - "$ref": "#/components/parameters/repo" - }, - { - "$ref": "#/components/parameters/per-page" - }, - { - "$ref": "#/components/parameters/page" - } - ], - "responses": { - "200": { - "description": "Response", - "content": { - "application/json": { - "schema": { - "type": "array", - "items": { - "$ref": "#/components/schemas/label" - } - }, - "examples": { - "default": { - "$ref": "#/components/examples/label-items" - } - } - } - }, - "headers": { - "Link": { - "$ref": "#/components/headers/link" - } - } - }, - "404": { - "$ref": "#/components/responses/not_found" - } - }, - "x-github": { - "githubCloudOnly": false, - "enabledForGitHubApps": true, - "category": "issues", - "subcategory": "labels" - } - } - } - } -} \ No newline at end of file diff --git a/dotnet/samples/GettingStarted/Agents/Agent_Step03.2_UsingFunctionTools_FromOpenAPI/Program.cs b/dotnet/samples/GettingStarted/Agents/Agent_Step03.2_UsingFunctionTools_FromOpenAPI/Program.cs deleted file mode 100644 index e61c9f845a..0000000000 --- a/dotnet/samples/GettingStarted/Agents/Agent_Step03.2_UsingFunctionTools_FromOpenAPI/Program.cs +++ /dev/null @@ -1,33 +0,0 @@ -// Copyright (c) Microsoft. All rights reserved. - -// This sample demonstrates how to use a ChatClientAgent with function tools provided via an OpenAPI spec. -// It uses functionality from Semantic Kernel to parse the OpenAPI spec and create function tools to use with the Agent Framework Agent. - -using Azure.AI.OpenAI; -using Azure.Identity; -using Microsoft.Agents.AI; -using Microsoft.Extensions.AI; -using Microsoft.SemanticKernel; -using Microsoft.SemanticKernel.Plugins.OpenApi; -using OpenAI; - -var endpoint = Environment.GetEnvironmentVariable("AZURE_OPENAI_ENDPOINT") ?? throw new InvalidOperationException("AZURE_OPENAI_ENDPOINT is not set."); -var deploymentName = Environment.GetEnvironmentVariable("AZURE_OPENAI_DEPLOYMENT_NAME") ?? "gpt-4o-mini"; - -// Load the OpenAPI Spec from a file. -KernelPlugin plugin = await OpenApiKernelPluginFactory.CreateFromOpenApiAsync("github", "OpenAPISpec.json"); - -// Convert the Semantic Kernel plugin to Agent Framework function tools. -// This requires a dummy Kernel instance, since KernelFunctions cannot execute without one. -Kernel kernel = new(); -List tools = plugin.Select(x => x.WithKernel(kernel)).Cast().ToList(); - -// Create the chat client and agent, and provide the OpenAPI function tools to the agent. -AIAgent agent = new AzureOpenAIClient( - new Uri(endpoint), - new AzureCliCredential()) - .GetChatClient(deploymentName) - .CreateAIAgent(instructions: "You are a helpful assistant", tools: tools); - -// Run the agent with the OpenAPI function tools. -Console.WriteLine(await agent.RunAsync("Please list the names, colors and descriptions of all the labels available in the microsoft/agent-framework repository on github.")); diff --git a/dotnet/samples/GettingStarted/Agents/Agent_Step13_Memory/Agent_Step13_Memory.csproj b/dotnet/samples/GettingStarted/Agents/Agent_Step03_UsingFunctionTools/Agent_Step03_UsingFunctionTools.csproj similarity index 91% rename from dotnet/samples/GettingStarted/Agents/Agent_Step13_Memory/Agent_Step13_Memory.csproj rename to dotnet/samples/GettingStarted/Agents/Agent_Step03_UsingFunctionTools/Agent_Step03_UsingFunctionTools.csproj index 8298cfe6e8..0f9de7c359 100644 --- a/dotnet/samples/GettingStarted/Agents/Agent_Step13_Memory/Agent_Step13_Memory.csproj +++ b/dotnet/samples/GettingStarted/Agents/Agent_Step03_UsingFunctionTools/Agent_Step03_UsingFunctionTools.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable diff --git a/dotnet/samples/GettingStarted/Agents/Agent_Step03.1_UsingFunctionTools/Program.cs b/dotnet/samples/GettingStarted/Agents/Agent_Step03_UsingFunctionTools/Program.cs similarity index 98% rename from dotnet/samples/GettingStarted/Agents/Agent_Step03.1_UsingFunctionTools/Program.cs rename to dotnet/samples/GettingStarted/Agents/Agent_Step03_UsingFunctionTools/Program.cs index 48a6378e1f..ae41572cc2 100644 --- a/dotnet/samples/GettingStarted/Agents/Agent_Step03.1_UsingFunctionTools/Program.cs +++ b/dotnet/samples/GettingStarted/Agents/Agent_Step03_UsingFunctionTools/Program.cs @@ -8,7 +8,7 @@ using Azure.Identity; using Microsoft.Agents.AI; using Microsoft.Extensions.AI; -using OpenAI; +using OpenAI.Chat; var endpoint = Environment.GetEnvironmentVariable("AZURE_OPENAI_ENDPOINT") ?? throw new InvalidOperationException("AZURE_OPENAI_ENDPOINT is not set."); var deploymentName = Environment.GetEnvironmentVariable("AZURE_OPENAI_DEPLOYMENT_NAME") ?? "gpt-4o-mini"; diff --git a/dotnet/samples/GettingStarted/Agents/Agent_Step04_UsingFunctionToolsWithApprovals/Agent_Step04_UsingFunctionToolsWithApprovals.csproj b/dotnet/samples/GettingStarted/Agents/Agent_Step04_UsingFunctionToolsWithApprovals/Agent_Step04_UsingFunctionToolsWithApprovals.csproj index 8298cfe6e8..0f9de7c359 100644 --- a/dotnet/samples/GettingStarted/Agents/Agent_Step04_UsingFunctionToolsWithApprovals/Agent_Step04_UsingFunctionToolsWithApprovals.csproj +++ b/dotnet/samples/GettingStarted/Agents/Agent_Step04_UsingFunctionToolsWithApprovals/Agent_Step04_UsingFunctionToolsWithApprovals.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable diff --git a/dotnet/samples/GettingStarted/Agents/Agent_Step04_UsingFunctionToolsWithApprovals/Program.cs b/dotnet/samples/GettingStarted/Agents/Agent_Step04_UsingFunctionToolsWithApprovals/Program.cs index 41ea8a5c92..be2a4801ae 100644 --- a/dotnet/samples/GettingStarted/Agents/Agent_Step04_UsingFunctionToolsWithApprovals/Program.cs +++ b/dotnet/samples/GettingStarted/Agents/Agent_Step04_UsingFunctionToolsWithApprovals/Program.cs @@ -10,7 +10,8 @@ using Azure.Identity; using Microsoft.Agents.AI; using Microsoft.Extensions.AI; -using OpenAI; +using OpenAI.Chat; +using ChatMessage = Microsoft.Extensions.AI.ChatMessage; var endpoint = Environment.GetEnvironmentVariable("AZURE_OPENAI_ENDPOINT") ?? throw new InvalidOperationException("AZURE_OPENAI_ENDPOINT is not set."); var deploymentName = Environment.GetEnvironmentVariable("AZURE_OPENAI_DEPLOYMENT_NAME") ?? "gpt-4o-mini"; diff --git a/dotnet/samples/GettingStarted/Agents/Agent_Step05_StructuredOutput/Agent_Step05_StructuredOutput.csproj b/dotnet/samples/GettingStarted/Agents/Agent_Step05_StructuredOutput/Agent_Step05_StructuredOutput.csproj index 8298cfe6e8..0f9de7c359 100644 --- a/dotnet/samples/GettingStarted/Agents/Agent_Step05_StructuredOutput/Agent_Step05_StructuredOutput.csproj +++ b/dotnet/samples/GettingStarted/Agents/Agent_Step05_StructuredOutput/Agent_Step05_StructuredOutput.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable diff --git a/dotnet/samples/GettingStarted/Agents/Agent_Step05_StructuredOutput/Program.cs b/dotnet/samples/GettingStarted/Agents/Agent_Step05_StructuredOutput/Program.cs index b18d8e2d84..3b923069f4 100644 --- a/dotnet/samples/GettingStarted/Agents/Agent_Step05_StructuredOutput/Program.cs +++ b/dotnet/samples/GettingStarted/Agents/Agent_Step05_StructuredOutput/Program.cs @@ -8,7 +8,6 @@ using Azure.AI.OpenAI; using Azure.Identity; using Microsoft.Agents.AI; -using OpenAI; using OpenAI.Chat; using SampleApp; @@ -22,7 +21,7 @@ .GetChatClient(deploymentName); // Create the ChatClientAgent with the specified name and instructions. -ChatClientAgent agent = chatClient.CreateAIAgent(new ChatClientAgentOptions(name: "HelpfulAssistant", instructions: "You are a helpful assistant.")); +ChatClientAgent agent = chatClient.CreateAIAgent(name: "HelpfulAssistant", instructions: "You are a helpful assistant."); // Set PersonInfo as the type parameter of RunAsync method to specify the expected structured output from the agent and invoke the agent with some unstructured input. AgentRunResponse response = await agent.RunAsync("Please provide information about John Smith, who is a 35-year-old software engineer."); @@ -34,12 +33,10 @@ Console.WriteLine($"Occupation: {response.Result.Occupation}"); // Create the ChatClientAgent with the specified name, instructions, and expected structured output the agent should produce. -ChatClientAgent agentWithPersonInfo = chatClient.CreateAIAgent(new ChatClientAgentOptions(name: "HelpfulAssistant", instructions: "You are a helpful assistant.") +ChatClientAgent agentWithPersonInfo = chatClient.CreateAIAgent(new ChatClientAgentOptions() { - ChatOptions = new() - { - ResponseFormat = Microsoft.Extensions.AI.ChatResponseFormat.ForJsonSchema() - } + Name = "HelpfulAssistant", + ChatOptions = new() { Instructions = "You are a helpful assistant.", ResponseFormat = Microsoft.Extensions.AI.ChatResponseFormat.ForJsonSchema() } }); // Invoke the agent with some unstructured input while streaming, to extract the structured information from. diff --git a/dotnet/samples/GettingStarted/Agents/Agent_Step06_PersistedConversations/Agent_Step06_PersistedConversations.csproj b/dotnet/samples/GettingStarted/Agents/Agent_Step06_PersistedConversations/Agent_Step06_PersistedConversations.csproj index 8298cfe6e8..0f9de7c359 100644 --- a/dotnet/samples/GettingStarted/Agents/Agent_Step06_PersistedConversations/Agent_Step06_PersistedConversations.csproj +++ b/dotnet/samples/GettingStarted/Agents/Agent_Step06_PersistedConversations/Agent_Step06_PersistedConversations.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable diff --git a/dotnet/samples/GettingStarted/Agents/Agent_Step06_PersistedConversations/Program.cs b/dotnet/samples/GettingStarted/Agents/Agent_Step06_PersistedConversations/Program.cs index 1ffe3c9993..5d3247b69c 100644 --- a/dotnet/samples/GettingStarted/Agents/Agent_Step06_PersistedConversations/Program.cs +++ b/dotnet/samples/GettingStarted/Agents/Agent_Step06_PersistedConversations/Program.cs @@ -6,7 +6,7 @@ using Azure.AI.OpenAI; using Azure.Identity; using Microsoft.Agents.AI; -using OpenAI; +using OpenAI.Chat; var endpoint = Environment.GetEnvironmentVariable("AZURE_OPENAI_ENDPOINT") ?? throw new InvalidOperationException("AZURE_OPENAI_ENDPOINT is not set."); var deploymentName = Environment.GetEnvironmentVariable("AZURE_OPENAI_DEPLOYMENT_NAME") ?? "gpt-4o-mini"; @@ -32,7 +32,7 @@ await File.WriteAllTextAsync(tempFilePath, JsonSerializer.Serialize(serializedThread)); // Load the serialized thread from the temporary file (for demonstration purposes). -JsonElement reloadedSerializedThread = JsonSerializer.Deserialize(await File.ReadAllTextAsync(tempFilePath)); +JsonElement reloadedSerializedThread = JsonElement.Parse(await File.ReadAllTextAsync(tempFilePath)); // Deserialize the thread state after loading from storage. AgentThread resumedThread = agent.DeserializeThread(reloadedSerializedThread); diff --git a/dotnet/samples/GettingStarted/Agents/Agent_Step07_3rdPartyThreadStorage/Agent_Step07_3rdPartyThreadStorage.csproj b/dotnet/samples/GettingStarted/Agents/Agent_Step07_3rdPartyThreadStorage/Agent_Step07_3rdPartyThreadStorage.csproj index 1caf270c49..860089b621 100644 --- a/dotnet/samples/GettingStarted/Agents/Agent_Step07_3rdPartyThreadStorage/Agent_Step07_3rdPartyThreadStorage.csproj +++ b/dotnet/samples/GettingStarted/Agents/Agent_Step07_3rdPartyThreadStorage/Agent_Step07_3rdPartyThreadStorage.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable @@ -13,7 +13,6 @@ - diff --git a/dotnet/samples/GettingStarted/Agents/Agent_Step07_3rdPartyThreadStorage/Program.cs b/dotnet/samples/GettingStarted/Agents/Agent_Step07_3rdPartyThreadStorage/Program.cs index 8986734972..e9794e871a 100644 --- a/dotnet/samples/GettingStarted/Agents/Agent_Step07_3rdPartyThreadStorage/Program.cs +++ b/dotnet/samples/GettingStarted/Agents/Agent_Step07_3rdPartyThreadStorage/Program.cs @@ -11,8 +11,9 @@ using Microsoft.Extensions.AI; using Microsoft.Extensions.VectorData; using Microsoft.SemanticKernel.Connectors.InMemory; -using OpenAI; +using OpenAI.Chat; using SampleApp; +using ChatMessage = Microsoft.Extensions.AI.ChatMessage; var endpoint = Environment.GetEnvironmentVariable("AZURE_OPENAI_ENDPOINT") ?? throw new InvalidOperationException("AZURE_OPENAI_ENDPOINT is not set."); var deploymentName = Environment.GetEnvironmentVariable("AZURE_OPENAI_DEPLOYMENT_NAME") ?? "gpt-4o-mini"; @@ -28,7 +29,7 @@ .GetChatClient(deploymentName) .CreateAIAgent(new ChatClientAgentOptions { - Instructions = "You are good at telling jokes.", + ChatOptions = new() { Instructions = "You are good at telling jokes." }, Name = "Joker", ChatMessageStoreFactory = ctx => { diff --git a/dotnet/samples/GettingStarted/Agents/Agent_Step08_Observability/Agent_Step08_Observability.csproj b/dotnet/samples/GettingStarted/Agents/Agent_Step08_Observability/Agent_Step08_Observability.csproj index 980e282641..1a618d660a 100644 --- a/dotnet/samples/GettingStarted/Agents/Agent_Step08_Observability/Agent_Step08_Observability.csproj +++ b/dotnet/samples/GettingStarted/Agents/Agent_Step08_Observability/Agent_Step08_Observability.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable diff --git a/dotnet/samples/GettingStarted/Agents/Agent_Step08_Observability/Program.cs b/dotnet/samples/GettingStarted/Agents/Agent_Step08_Observability/Program.cs index c48242f5ca..e43e80d664 100644 --- a/dotnet/samples/GettingStarted/Agents/Agent_Step08_Observability/Program.cs +++ b/dotnet/samples/GettingStarted/Agents/Agent_Step08_Observability/Program.cs @@ -6,7 +6,7 @@ using Azure.Identity; using Azure.Monitor.OpenTelemetry.Exporter; using Microsoft.Agents.AI; -using OpenAI; +using OpenAI.Chat; using OpenTelemetry; using OpenTelemetry.Trace; diff --git a/dotnet/samples/GettingStarted/Agents/Agent_Step09_DependencyInjection/Agent_Step09_DependencyInjection.csproj b/dotnet/samples/GettingStarted/Agents/Agent_Step09_DependencyInjection/Agent_Step09_DependencyInjection.csproj index b0890e1817..0aaa471260 100644 --- a/dotnet/samples/GettingStarted/Agents/Agent_Step09_DependencyInjection/Agent_Step09_DependencyInjection.csproj +++ b/dotnet/samples/GettingStarted/Agents/Agent_Step09_DependencyInjection/Agent_Step09_DependencyInjection.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable diff --git a/dotnet/samples/GettingStarted/Agents/Agent_Step09_DependencyInjection/Program.cs b/dotnet/samples/GettingStarted/Agents/Agent_Step09_DependencyInjection/Program.cs index 894c034eb0..d1b75d2fe5 100644 --- a/dotnet/samples/GettingStarted/Agents/Agent_Step09_DependencyInjection/Program.cs +++ b/dotnet/samples/GettingStarted/Agents/Agent_Step09_DependencyInjection/Program.cs @@ -18,8 +18,7 @@ HostApplicationBuilder builder = Host.CreateApplicationBuilder(args); // Add agent options to the service collection. -builder.Services.AddSingleton( - new ChatClientAgentOptions(instructions: "You are good at telling jokes.", name: "Joker")); +builder.Services.AddSingleton(new ChatClientAgentOptions() { Name = "Joker", ChatOptions = new() { Instructions = "You are good at telling jokes." } }); // Add a chat client to the service collection. builder.Services.AddKeyedChatClient("AzureOpenAI", (sp) => new AzureOpenAIClient( diff --git a/dotnet/samples/GettingStarted/Agents/Agent_Step10_AsMcpTool/Agent_Step10_AsMcpTool.csproj b/dotnet/samples/GettingStarted/Agents/Agent_Step10_AsMcpTool/Agent_Step10_AsMcpTool.csproj index 1fb367c044..db776afd1e 100644 --- a/dotnet/samples/GettingStarted/Agents/Agent_Step10_AsMcpTool/Agent_Step10_AsMcpTool.csproj +++ b/dotnet/samples/GettingStarted/Agents/Agent_Step10_AsMcpTool/Agent_Step10_AsMcpTool.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable @@ -14,7 +14,6 @@ - diff --git a/dotnet/samples/GettingStarted/Agents/Agent_Step11_UsingImages/Agent_Step11_UsingImages.csproj b/dotnet/samples/GettingStarted/Agents/Agent_Step11_UsingImages/Agent_Step11_UsingImages.csproj index 7e9e70c763..73a41005f1 100644 --- a/dotnet/samples/GettingStarted/Agents/Agent_Step11_UsingImages/Agent_Step11_UsingImages.csproj +++ b/dotnet/samples/GettingStarted/Agents/Agent_Step11_UsingImages/Agent_Step11_UsingImages.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable diff --git a/dotnet/samples/GettingStarted/Agents/Agent_Step11_UsingImages/Program.cs b/dotnet/samples/GettingStarted/Agents/Agent_Step11_UsingImages/Program.cs index 8e8c5701ad..f534e4edd7 100644 --- a/dotnet/samples/GettingStarted/Agents/Agent_Step11_UsingImages/Program.cs +++ b/dotnet/samples/GettingStarted/Agents/Agent_Step11_UsingImages/Program.cs @@ -5,7 +5,8 @@ using Azure.AI.OpenAI; using Azure.Identity; using Microsoft.Extensions.AI; -using OpenAI; +using OpenAI.Chat; +using ChatMessage = Microsoft.Extensions.AI.ChatMessage; var endpoint = Environment.GetEnvironmentVariable("AZURE_OPENAI_ENDPOINT") ?? throw new InvalidOperationException("AZURE_OPENAI_ENDPOINT is not set."); var deploymentName = System.Environment.GetEnvironmentVariable("AZURE_OPENAI_DEPLOYMENT_NAME") ?? "gpt-4o"; diff --git a/dotnet/samples/GettingStarted/Agents/Agent_Step12_AsFunctionTool/Agent_Step12_AsFunctionTool.csproj b/dotnet/samples/GettingStarted/Agents/Agent_Step12_AsFunctionTool/Agent_Step12_AsFunctionTool.csproj index 21c8d9e49e..2660090404 100644 --- a/dotnet/samples/GettingStarted/Agents/Agent_Step12_AsFunctionTool/Agent_Step12_AsFunctionTool.csproj +++ b/dotnet/samples/GettingStarted/Agents/Agent_Step12_AsFunctionTool/Agent_Step12_AsFunctionTool.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable diff --git a/dotnet/samples/GettingStarted/Agents/Agent_Step12_AsFunctionTool/Program.cs b/dotnet/samples/GettingStarted/Agents/Agent_Step12_AsFunctionTool/Program.cs index cce53ef3c0..5e37ff4039 100644 --- a/dotnet/samples/GettingStarted/Agents/Agent_Step12_AsFunctionTool/Program.cs +++ b/dotnet/samples/GettingStarted/Agents/Agent_Step12_AsFunctionTool/Program.cs @@ -7,7 +7,7 @@ using Azure.Identity; using Microsoft.Agents.AI; using Microsoft.Extensions.AI; -using OpenAI; +using OpenAI.Chat; var endpoint = Environment.GetEnvironmentVariable("AZURE_OPENAI_ENDPOINT") ?? throw new InvalidOperationException("AZURE_OPENAI_ENDPOINT is not set."); var deploymentName = Environment.GetEnvironmentVariable("AZURE_OPENAI_DEPLOYMENT_NAME") ?? "gpt-4o-mini"; diff --git a/dotnet/samples/GettingStarted/Agents/Agent_Step20_BackgroundResponsesWithToolsAndPersistence/Agent_Step20_BackgroundResponsesWithToolsAndPersistence.csproj b/dotnet/samples/GettingStarted/Agents/Agent_Step13_BackgroundResponsesWithToolsAndPersistence/Agent_Step13_BackgroundResponsesWithToolsAndPersistence.csproj similarity index 90% rename from dotnet/samples/GettingStarted/Agents/Agent_Step20_BackgroundResponsesWithToolsAndPersistence/Agent_Step20_BackgroundResponsesWithToolsAndPersistence.csproj rename to dotnet/samples/GettingStarted/Agents/Agent_Step13_BackgroundResponsesWithToolsAndPersistence/Agent_Step13_BackgroundResponsesWithToolsAndPersistence.csproj index 4735f4a7a0..29fab5f992 100644 --- a/dotnet/samples/GettingStarted/Agents/Agent_Step20_BackgroundResponsesWithToolsAndPersistence/Agent_Step20_BackgroundResponsesWithToolsAndPersistence.csproj +++ b/dotnet/samples/GettingStarted/Agents/Agent_Step13_BackgroundResponsesWithToolsAndPersistence/Agent_Step13_BackgroundResponsesWithToolsAndPersistence.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable diff --git a/dotnet/samples/GettingStarted/Agents/Agent_Step20_BackgroundResponsesWithToolsAndPersistence/Program.cs b/dotnet/samples/GettingStarted/Agents/Agent_Step13_BackgroundResponsesWithToolsAndPersistence/Program.cs similarity index 93% rename from dotnet/samples/GettingStarted/Agents/Agent_Step20_BackgroundResponsesWithToolsAndPersistence/Program.cs rename to dotnet/samples/GettingStarted/Agents/Agent_Step13_BackgroundResponsesWithToolsAndPersistence/Program.cs index f2a3bdf5c0..41493f6d79 100644 --- a/dotnet/samples/GettingStarted/Agents/Agent_Step20_BackgroundResponsesWithToolsAndPersistence/Program.cs +++ b/dotnet/samples/GettingStarted/Agents/Agent_Step13_BackgroundResponsesWithToolsAndPersistence/Program.cs @@ -12,7 +12,7 @@ using Azure.Identity; using Microsoft.Agents.AI; using Microsoft.Extensions.AI; -using OpenAI; +using OpenAI.Responses; var endpoint = Environment.GetEnvironmentVariable("AZURE_OPENAI_ENDPOINT") ?? throw new InvalidOperationException("AZURE_OPENAI_ENDPOINT is not set."); var deploymentName = Environment.GetEnvironmentVariable("AZURE_OPENAI_DEPLOYMENT_NAME") ?? "gpt-5"; @@ -44,7 +44,7 @@ await Task.Delay(TimeSpan.FromSeconds(10)); - RestoreAgentState(agent, out thread, out object? continuationToken); + RestoreAgentState(agent, out thread, out ResponseContinuationToken? continuationToken); options.ContinuationToken = continuationToken; response = await agent.RunAsync(thread, options); @@ -52,19 +52,19 @@ Console.WriteLine(response.Text); -void PersistAgentState(AgentThread thread, object? continuationToken) +void PersistAgentState(AgentThread thread, ResponseContinuationToken? continuationToken) { stateStore["thread"] = thread.Serialize(); stateStore["continuationToken"] = JsonSerializer.SerializeToElement(continuationToken, AgentAbstractionsJsonUtilities.DefaultOptions.GetTypeInfo(typeof(ResponseContinuationToken))); } -void RestoreAgentState(AIAgent agent, out AgentThread thread, out object? continuationToken) +void RestoreAgentState(AIAgent agent, out AgentThread thread, out ResponseContinuationToken? continuationToken) { JsonElement serializedThread = stateStore["thread"] ?? throw new InvalidOperationException("No serialized thread found in state store."); JsonElement? serializedToken = stateStore["continuationToken"]; thread = agent.DeserializeThread(serializedThread); - continuationToken = serializedToken?.Deserialize(AgentAbstractionsJsonUtilities.DefaultOptions.GetTypeInfo(typeof(ResponseContinuationToken))); + continuationToken = (ResponseContinuationToken?)serializedToken?.Deserialize(AgentAbstractionsJsonUtilities.DefaultOptions.GetTypeInfo(typeof(ResponseContinuationToken))); } [Description("Researches relevant space facts and scientific information for writing a science fiction novel")] diff --git a/dotnet/samples/GettingStarted/Agents/Agent_Step20_BackgroundResponsesWithToolsAndPersistence/README.md b/dotnet/samples/GettingStarted/Agents/Agent_Step13_BackgroundResponsesWithToolsAndPersistence/README.md similarity index 98% rename from dotnet/samples/GettingStarted/Agents/Agent_Step20_BackgroundResponsesWithToolsAndPersistence/README.md rename to dotnet/samples/GettingStarted/Agents/Agent_Step13_BackgroundResponsesWithToolsAndPersistence/README.md index 146f418512..ca52e8afa3 100644 --- a/dotnet/samples/GettingStarted/Agents/Agent_Step20_BackgroundResponsesWithToolsAndPersistence/README.md +++ b/dotnet/samples/GettingStarted/Agents/Agent_Step13_BackgroundResponsesWithToolsAndPersistence/README.md @@ -14,7 +14,7 @@ For more information, see the [official documentation](https://learn.microsoft.c Before you begin, ensure you have the following prerequisites: -- .NET 8.0 SDK or later +- .NET 10 SDK or later - Azure OpenAI service endpoint and deployment configured - Azure CLI installed and authenticated (for Azure credential authentication) diff --git a/dotnet/samples/GettingStarted/Agents/Agent_Step14_Middleware/Agent_Step14_Middleware.csproj b/dotnet/samples/GettingStarted/Agents/Agent_Step14_Middleware/Agent_Step14_Middleware.csproj index 09beb78195..6582c30cd5 100644 --- a/dotnet/samples/GettingStarted/Agents/Agent_Step14_Middleware/Agent_Step14_Middleware.csproj +++ b/dotnet/samples/GettingStarted/Agents/Agent_Step14_Middleware/Agent_Step14_Middleware.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable diff --git a/dotnet/samples/GettingStarted/Agents/Agent_Step14_Middleware/Program.cs b/dotnet/samples/GettingStarted/Agents/Agent_Step14_Middleware/Program.cs index 28a50cc7d7..a0ca338297 100644 --- a/dotnet/samples/GettingStarted/Agents/Agent_Step14_Middleware/Program.cs +++ b/dotnet/samples/GettingStarted/Agents/Agent_Step14_Middleware/Program.cs @@ -154,10 +154,11 @@ static IList FilterMessages(IEnumerable messages) static string FilterPii(string content) { // Regex patterns for PII detection (simplified for demonstration) - Regex[] piiPatterns = [ + Regex[] piiPatterns = + [ new(@"\b\d{3}-\d{3}-\d{4}\b", RegexOptions.Compiled), // Phone number (e.g., 123-456-7890) - new(@"\b[\w\.-]+@[\w\.-]+\.\w+\b", RegexOptions.Compiled), // Email address - new(@"\b[A-Z][a-z]+\s[A-Z][a-z]+\b", RegexOptions.Compiled) // Full name (e.g., John Doe) + new(@"\b[\w\.-]+@[\w\.-]+\.\w+\b", RegexOptions.Compiled), // Email address + new(@"\b[A-Z][a-z]+\s[A-Z][a-z]+\b", RegexOptions.Compiled) // Full name (e.g., John Doe) ]; foreach (var pattern in piiPatterns) diff --git a/dotnet/samples/GettingStarted/Agents/Agent_Step15_Plugins/Agent_Step15_Plugins.csproj b/dotnet/samples/GettingStarted/Agents/Agent_Step15_Plugins/Agent_Step15_Plugins.csproj index c1cf0bf930..ae2f9ac194 100644 --- a/dotnet/samples/GettingStarted/Agents/Agent_Step15_Plugins/Agent_Step15_Plugins.csproj +++ b/dotnet/samples/GettingStarted/Agents/Agent_Step15_Plugins/Agent_Step15_Plugins.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable diff --git a/dotnet/samples/GettingStarted/Agents/Agent_Step15_Plugins/Program.cs b/dotnet/samples/GettingStarted/Agents/Agent_Step15_Plugins/Program.cs index 7284efcc42..38cd20b8d6 100644 --- a/dotnet/samples/GettingStarted/Agents/Agent_Step15_Plugins/Program.cs +++ b/dotnet/samples/GettingStarted/Agents/Agent_Step15_Plugins/Program.cs @@ -14,7 +14,7 @@ using Microsoft.Agents.AI; using Microsoft.Extensions.AI; using Microsoft.Extensions.DependencyInjection; -using OpenAI; +using OpenAI.Chat; var endpoint = Environment.GetEnvironmentVariable("AZURE_OPENAI_ENDPOINT") ?? throw new InvalidOperationException("AZURE_OPENAI_ENDPOINT is not set."); var deploymentName = Environment.GetEnvironmentVariable("AZURE_OPENAI_DEPLOYMENT_NAME") ?? "gpt-4o-mini"; diff --git a/dotnet/samples/GettingStarted/Agents/Agent_Step16_ChatReduction/Agent_Step16_ChatReduction.csproj b/dotnet/samples/GettingStarted/Agents/Agent_Step16_ChatReduction/Agent_Step16_ChatReduction.csproj index 8298cfe6e8..0f9de7c359 100644 --- a/dotnet/samples/GettingStarted/Agents/Agent_Step16_ChatReduction/Agent_Step16_ChatReduction.csproj +++ b/dotnet/samples/GettingStarted/Agents/Agent_Step16_ChatReduction/Agent_Step16_ChatReduction.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable diff --git a/dotnet/samples/GettingStarted/Agents/Agent_Step16_ChatReduction/Program.cs b/dotnet/samples/GettingStarted/Agents/Agent_Step16_ChatReduction/Program.cs index 590b5308d5..decf0de25a 100644 --- a/dotnet/samples/GettingStarted/Agents/Agent_Step16_ChatReduction/Program.cs +++ b/dotnet/samples/GettingStarted/Agents/Agent_Step16_ChatReduction/Program.cs @@ -9,7 +9,8 @@ using Azure.Identity; using Microsoft.Agents.AI; using Microsoft.Extensions.AI; -using OpenAI; +using OpenAI.Chat; +using ChatMessage = Microsoft.Extensions.AI.ChatMessage; var endpoint = Environment.GetEnvironmentVariable("AZURE_OPENAI_ENDPOINT") ?? throw new InvalidOperationException("AZURE_OPENAI_ENDPOINT is not set."); var deploymentName = Environment.GetEnvironmentVariable("AZURE_OPENAI_DEPLOYMENT_NAME") ?? "gpt-4o-mini"; @@ -21,7 +22,7 @@ .GetChatClient(deploymentName) .CreateAIAgent(new ChatClientAgentOptions { - Instructions = "You are good at telling jokes.", + ChatOptions = new() { Instructions = "You are good at telling jokes." }, Name = "Joker", ChatMessageStoreFactory = ctx => new InMemoryChatMessageStore(new MessageCountingChatReducer(2), ctx.SerializedState, ctx.JsonSerializerOptions) }); diff --git a/dotnet/samples/GettingStarted/Agents/Agent_Step17_BackgroundResponses/Agent_Step17_BackgroundResponses.csproj b/dotnet/samples/GettingStarted/Agents/Agent_Step17_BackgroundResponses/Agent_Step17_BackgroundResponses.csproj index c5b2ae56a6..1c95b4af25 100644 --- a/dotnet/samples/GettingStarted/Agents/Agent_Step17_BackgroundResponses/Agent_Step17_BackgroundResponses.csproj +++ b/dotnet/samples/GettingStarted/Agents/Agent_Step17_BackgroundResponses/Agent_Step17_BackgroundResponses.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable diff --git a/dotnet/samples/GettingStarted/Agents/Agent_Step17_BackgroundResponses/Program.cs b/dotnet/samples/GettingStarted/Agents/Agent_Step17_BackgroundResponses/Program.cs index 456d968c34..510a5dfbd0 100644 --- a/dotnet/samples/GettingStarted/Agents/Agent_Step17_BackgroundResponses/Program.cs +++ b/dotnet/samples/GettingStarted/Agents/Agent_Step17_BackgroundResponses/Program.cs @@ -5,7 +5,7 @@ using Azure.AI.OpenAI; using Azure.Identity; using Microsoft.Agents.AI; -using OpenAI; +using OpenAI.Responses; var endpoint = Environment.GetEnvironmentVariable("AZURE_OPENAI_ENDPOINT") ?? throw new InvalidOperationException("AZURE_OPENAI_ENDPOINT is not set."); var deploymentName = Environment.GetEnvironmentVariable("AZURE_OPENAI_DEPLOYMENT_NAME") ?? "gpt-4o-mini"; diff --git a/dotnet/samples/GettingStarted/Agents/Agent_Step17_BackgroundResponses/README.md b/dotnet/samples/GettingStarted/Agents/Agent_Step17_BackgroundResponses/README.md index 5b7df74ca9..e898733bc3 100644 --- a/dotnet/samples/GettingStarted/Agents/Agent_Step17_BackgroundResponses/README.md +++ b/dotnet/samples/GettingStarted/Agents/Agent_Step17_BackgroundResponses/README.md @@ -13,7 +13,7 @@ For more information, see the [official documentation](https://learn.microsoft.c Before you begin, ensure you have the following prerequisites: -- .NET 8.0 SDK or later +- .NET 10 SDK or later - Azure OpenAI service endpoint and deployment configured - Azure CLI installed and authenticated (for Azure credential authentication) diff --git a/dotnet/samples/Catalog/DeepResearchAgent/DeepResearchAgent.csproj b/dotnet/samples/GettingStarted/Agents/Agent_Step18_DeepResearch/Agent_Step18_DeepResearch.csproj similarity index 66% rename from dotnet/samples/Catalog/DeepResearchAgent/DeepResearchAgent.csproj rename to dotnet/samples/GettingStarted/Agents/Agent_Step18_DeepResearch/Agent_Step18_DeepResearch.csproj index 7ae71d83de..d40e93232b 100644 --- a/dotnet/samples/Catalog/DeepResearchAgent/DeepResearchAgent.csproj +++ b/dotnet/samples/GettingStarted/Agents/Agent_Step18_DeepResearch/Agent_Step18_DeepResearch.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable @@ -14,7 +14,7 @@ - + diff --git a/dotnet/samples/Catalog/DeepResearchAgent/Program.cs b/dotnet/samples/GettingStarted/Agents/Agent_Step18_DeepResearch/Program.cs similarity index 100% rename from dotnet/samples/Catalog/DeepResearchAgent/Program.cs rename to dotnet/samples/GettingStarted/Agents/Agent_Step18_DeepResearch/Program.cs diff --git a/dotnet/samples/Catalog/DeepResearchAgent/README.md b/dotnet/samples/GettingStarted/Agents/Agent_Step18_DeepResearch/README.md similarity index 100% rename from dotnet/samples/Catalog/DeepResearchAgent/README.md rename to dotnet/samples/GettingStarted/Agents/Agent_Step18_DeepResearch/README.md diff --git a/dotnet/samples/GettingStarted/Agents/Agent_Step19_Declarative/Agent_Step19_Declarative.csproj b/dotnet/samples/GettingStarted/Agents/Agent_Step19_Declarative/Agent_Step19_Declarative.csproj new file mode 100644 index 0000000000..550e1f22cb --- /dev/null +++ b/dotnet/samples/GettingStarted/Agents/Agent_Step19_Declarative/Agent_Step19_Declarative.csproj @@ -0,0 +1,25 @@ + + + + Exe + net10.0 + + enable + enable + + + + + + + + + + + + + + + + + diff --git a/dotnet/samples/GettingStarted/Agents/Agent_Step19_Declarative/Program.cs b/dotnet/samples/GettingStarted/Agents/Agent_Step19_Declarative/Program.cs new file mode 100644 index 0000000000..1fc985b3bb --- /dev/null +++ b/dotnet/samples/GettingStarted/Agents/Agent_Step19_Declarative/Program.cs @@ -0,0 +1,54 @@ +// Copyright (c) Microsoft. All rights reserved. + +// This sample shows how to create an agent from a YAML based declarative representation. + +using Azure.AI.OpenAI; +using Azure.Identity; +using Microsoft.Agents.AI; +using Microsoft.Extensions.AI; + +var endpoint = Environment.GetEnvironmentVariable("AZURE_OPENAI_ENDPOINT") ?? throw new InvalidOperationException("AZURE_OPENAI_ENDPOINT is not set."); +var deploymentName = Environment.GetEnvironmentVariable("AZURE_OPENAI_DEPLOYMENT_NAME") ?? "gpt-4o-mini"; + +// Create the chat client +IChatClient chatClient = new AzureOpenAIClient( + new Uri(endpoint), + new AzureCliCredential()) + .GetChatClient(deploymentName) + .AsIChatClient(); + +// Define the agent using a YAML definition. +var text = + """ + kind: Prompt + name: Assistant + description: Helpful assistant + instructions: You are a helpful assistant. You answer questions in the language specified by the user. You return your answers in a JSON format. + model: + options: + temperature: 0.9 + topP: 0.95 + outputSchema: + properties: + language: + type: string + required: true + description: The language of the answer. + answer: + type: string + required: true + description: The answer text. + """; + +// Create the agent from the YAML definition. +var agentFactory = new ChatClientPromptAgentFactory(chatClient); +var agent = await agentFactory.CreateFromYamlAsync(text); + +// Invoke the agent and output the text result. +Console.WriteLine(await agent!.RunAsync("Tell me a joke about a pirate in English.")); + +// Invoke the agent with streaming support. +await foreach (var update in agent!.RunStreamingAsync("Tell me a joke about a pirate in French.")) +{ + Console.WriteLine(update); +} diff --git a/dotnet/samples/GettingStarted/Agents/README.md b/dotnet/samples/GettingStarted/Agents/README.md index 562b6b2500..d023d6455c 100644 --- a/dotnet/samples/GettingStarted/Agents/README.md +++ b/dotnet/samples/GettingStarted/Agents/README.md @@ -13,7 +13,7 @@ see the [How to create an agent for each provider](../AgentProviders/README.md) Before you begin, ensure you have the following prerequisites: -- .NET 8.0 SDK or later +- .NET 10 SDK or later - Azure OpenAI service endpoint and deployment configured - Azure CLI installed and authenticated (for Azure credential authentication) - User has the `Cognitive Services OpenAI Contributor` role for the Azure OpenAI resource. @@ -28,8 +28,8 @@ Before you begin, ensure you have the following prerequisites: |---|---| |[Running a simple agent](./Agent_Step01_Running/)|This sample demonstrates how to create and run a basic agent with instructions| |[Multi-turn conversation with a simple agent](./Agent_Step02_MultiturnConversation/)|This sample demonstrates how to implement a multi-turn conversation with a simple agent| -|[Using function tools with a simple agent](./Agent_Step03.1_UsingFunctionTools/)|This sample demonstrates how to use function tools with a simple agent| -|[Using OpenAPI function tools with a simple agent](./Agent_Step03.2_UsingFunctionTools_FromOpenAPI/)|This sample demonstrates how to create function tools from an OpenAPI spec and use them with a simple agent| +|[Using function tools with a simple agent](./Agent_Step03_UsingFunctionTools/)|This sample demonstrates how to use function tools with a simple agent| +|[Using OpenAPI function tools with a simple agent](https://github.com/microsoft/semantic-kernel/tree/main/dotnet/samples/AgentFrameworkMigration/AzureOpenAI/Step04_ToolCall_WithOpenAPI)|This sample demonstrates how to create function tools from an OpenAPI spec and use them with a simple agent (note that this sample is in the Semantic Kernel repository)| |[Using function tools with approvals](./Agent_Step04_UsingFunctionToolsWithApprovals/)|This sample demonstrates how to use function tools where approvals require human in the loop approvals before execution| |[Structured output with a simple agent](./Agent_Step05_StructuredOutput/)|This sample demonstrates how to use structured output with a simple agent| |[Persisted conversations with a simple agent](./Agent_Step06_PersistedConversations/)|This sample demonstrates how to persist conversations and reload them later. This is useful for cases where an agent is hosted in a stateless service| @@ -39,14 +39,13 @@ Before you begin, ensure you have the following prerequisites: |[Exposing a simple agent as MCP tool](./Agent_Step10_AsMcpTool/)|This sample demonstrates how to expose an agent as an MCP tool| |[Using images with a simple agent](./Agent_Step11_UsingImages/)|This sample demonstrates how to use image multi-modality with an AI agent| |[Exposing a simple agent as a function tool](./Agent_Step12_AsFunctionTool/)|This sample demonstrates how to expose an agent as a function tool| -|[Using memory with an agent](./Agent_Step13_Memory/)|This sample demonstrates how to create a simple memory component and use it with an agent| +|[Background responses with tools and persistence](./Agent_Step13_BackgroundResponsesWithToolsAndPersistence/)|This sample demonstrates advanced background response scenarios including function calling during background operations and state persistence| |[Using middleware with an agent](./Agent_Step14_Middleware/)|This sample demonstrates how to use middleware with an agent| |[Using plugins with an agent](./Agent_Step15_Plugins/)|This sample demonstrates how to use plugins with an agent| |[Reducing chat history size](./Agent_Step16_ChatReduction/)|This sample demonstrates how to reduce the chat history to constrain its size, where chat history is maintained locally| |[Background responses](./Agent_Step17_BackgroundResponses/)|This sample demonstrates how to use background responses for long-running operations with polling and resumption support| -|[Adding RAG with text search](./Agent_Step18_TextSearchRag/)|This sample demonstrates how to enrich agent responses with retrieval augmented generation using the text search provider| -|[Using Mem0-backed memory](./Agent_Step19_Mem0Provider/)|This sample demonstrates how to use the Mem0Provider to persist and recall memories across conversations| -|[Background responses with tools and persistence](./Agent_Step20_BackgroundResponsesWithToolsAndPersistence/)|This sample demonstrates advanced background response scenarios including function calling during background operations and state persistence| +|[Deep research with an agent](./Agent_Step18_DeepResearch/)|This sample demonstrates how to use the Deep Research Tool to perform comprehensive research on complex topics| +|[Declarative agent](./Agent_Step19_Declarative/)|This sample demonstrates how to declaratively define an agent.| ## Running the samples from the console diff --git a/dotnet/samples/GettingStarted/DeclarativeAgents/ChatClient/DeclarativeChatClientAgents.csproj b/dotnet/samples/GettingStarted/DeclarativeAgents/ChatClient/DeclarativeChatClientAgents.csproj new file mode 100644 index 0000000000..0fc316acac --- /dev/null +++ b/dotnet/samples/GettingStarted/DeclarativeAgents/ChatClient/DeclarativeChatClientAgents.csproj @@ -0,0 +1,25 @@ + + + + Exe + net10.0 + + enable + enable + + + + + + + + + + + + + + + + + diff --git a/dotnet/samples/GettingStarted/DeclarativeAgents/ChatClient/Program.cs b/dotnet/samples/GettingStarted/DeclarativeAgents/ChatClient/Program.cs new file mode 100644 index 0000000000..bed16f496a --- /dev/null +++ b/dotnet/samples/GettingStarted/DeclarativeAgents/ChatClient/Program.cs @@ -0,0 +1,55 @@ +// Copyright (c) Microsoft. All rights reserved. + +// This sample shows how to load an AI agent from a YAML file and process a prompt using Azure OpenAI as the backend. + +using System.ComponentModel; +using Azure.AI.OpenAI; +using Azure.Identity; +using Microsoft.Agents.AI; +using Microsoft.Extensions.AI; + +var endpoint = Environment.GetEnvironmentVariable("AZURE_OPENAI_ENDPOINT") ?? throw new InvalidOperationException("AZURE_OPENAI_ENDPOINT is not set."); +var deploymentName = Environment.GetEnvironmentVariable("AZURE_OPENAI_DEPLOYMENT_NAME") ?? "gpt-4o-mini"; + +// Create the chat client +IChatClient chatClient = new AzureOpenAIClient( + new Uri(endpoint), + new AzureCliCredential()) + .GetChatClient(deploymentName) + .AsIChatClient(); + +// Read command-line arguments +if (args.Length < 2) +{ + Console.WriteLine("Usage: DeclarativeAgents "); + Console.WriteLine(" : The path to the YAML file containing the agent definition"); + Console.WriteLine(" : The prompt to send to the agent"); + return; +} + +var yamlFilePath = args[0]; +var prompt = args[1]; + +// Verify the YAML file exists +if (!File.Exists(yamlFilePath)) +{ + Console.WriteLine($"Error: File not found: {yamlFilePath}"); + return; +} + +// Read the YAML content from the file +var text = await File.ReadAllTextAsync(yamlFilePath); + +// Example function tool that can be used by the agent. +[Description("Get the weather for a given location.")] +static string GetWeather( + [Description("The city and state, e.g. San Francisco, CA")] string location, + [Description("The unit of temperature. Possible values are 'celsius' and 'fahrenheit'.")] string unit) + => $"The weather in {location} is cloudy with a high of {(unit.Equals("celsius", StringComparison.Ordinal) ? "15°C" : "59°F")}."; + +// Create the agent from the YAML definition. +var agentFactory = new ChatClientPromptAgentFactory(chatClient, [AIFunctionFactory.Create(GetWeather, "GetWeather")]); +var agent = await agentFactory.CreateFromYamlAsync(text); + +// Invoke the agent and output the text result. +Console.WriteLine(await agent!.RunAsync(prompt)); diff --git a/dotnet/samples/GettingStarted/DeclarativeAgents/ChatClient/Properties/launchSettings.json b/dotnet/samples/GettingStarted/DeclarativeAgents/ChatClient/Properties/launchSettings.json new file mode 100644 index 0000000000..5ec486626c --- /dev/null +++ b/dotnet/samples/GettingStarted/DeclarativeAgents/ChatClient/Properties/launchSettings.json @@ -0,0 +1,12 @@ +{ + "profiles": { + "GetWeather": { + "commandName": "Project", + "commandLineArgs": "..\\..\\..\\..\\..\\..\\..\\..\\agent-samples\\chatclient\\GetWeather.yaml \"What is the weather in Cambridge, MA in °C?\"" + }, + "Assistant": { + "commandName": "Project", + "commandLineArgs": "..\\..\\..\\..\\..\\..\\..\\..\\agent-samples\\chatclient\\Assistant.yaml \"Tell me a joke about a pirate in Italian.\"" + } + } +} \ No newline at end of file diff --git a/dotnet/samples/GettingStarted/DevUI/DevUI_Step01_BasicUsage/DevUI_Step01_BasicUsage.csproj b/dotnet/samples/GettingStarted/DevUI/DevUI_Step01_BasicUsage/DevUI_Step01_BasicUsage.csproj index 8ae36b52e0..09037b5f1d 100644 --- a/dotnet/samples/GettingStarted/DevUI/DevUI_Step01_BasicUsage/DevUI_Step01_BasicUsage.csproj +++ b/dotnet/samples/GettingStarted/DevUI/DevUI_Step01_BasicUsage/DevUI_Step01_BasicUsage.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable DevUI_Step01_BasicUsage @@ -19,7 +19,6 @@ - diff --git a/dotnet/samples/GettingStarted/DevUI/DevUI_Step01_BasicUsage/Program.cs b/dotnet/samples/GettingStarted/DevUI/DevUI_Step01_BasicUsage/Program.cs index e2e6e6b727..7fded8c55b 100644 --- a/dotnet/samples/GettingStarted/DevUI/DevUI_Step01_BasicUsage/Program.cs +++ b/dotnet/samples/GettingStarted/DevUI/DevUI_Step01_BasicUsage/Program.cs @@ -2,10 +2,13 @@ // This sample demonstrates basic usage of the DevUI in an ASP.NET Core application with AI agents. +using System.ComponentModel; using Azure.AI.OpenAI; using Azure.Identity; +using Microsoft.Agents.AI; using Microsoft.Agents.AI.DevUI; using Microsoft.Agents.AI.Hosting; +using Microsoft.Agents.AI.Workflows; using Microsoft.Extensions.AI; namespace DevUI_Step01_BasicUsage; @@ -16,10 +19,11 @@ namespace DevUI_Step01_BasicUsage; /// /// This sample shows how to: /// 1. Set up Azure OpenAI as the chat client -/// 2. Register agents and workflows using the hosting packages -/// 3. Map the DevUI endpoint which automatically configures the middleware -/// 4. Map the dynamic OpenAI Responses API for Python DevUI compatibility -/// 5. Access the DevUI in a web browser +/// 2. Create function tools for agents to use +/// 3. Register agents and workflows using the hosting packages with tools +/// 4. Map the DevUI endpoint which automatically configures the middleware +/// 5. Map the dynamic OpenAI Responses API for Python DevUI compatibility +/// 6. Access the DevUI in a web browser /// /// The DevUI provides an interactive web interface for testing and debugging AI agents. /// DevUI assets are served from embedded resources within the assembly. @@ -48,26 +52,48 @@ private static void Main(string[] args) builder.Services.AddChatClient(chatClient); - // Register sample agents - builder.AddAIAgent("assistant", "You are a helpful assistant. Answer questions concisely and accurately."); + // Define some example tools + [Description("Get the weather for a given location.")] + static string GetWeather([Description("The location to get the weather for.")] string location) + => $"The weather in {location} is cloudy with a high of 15°C."; + + [Description("Calculate the sum of two numbers.")] + static double Add([Description("The first number.")] double a, [Description("The second number.")] double b) + => a + b; + + [Description("Get the current time.")] + static string GetCurrentTime() + => DateTime.Now.ToString("HH:mm:ss"); + + // Register sample agents with tools + builder.AddAIAgent("assistant", "You are a helpful assistant. Answer questions concisely and accurately.") + .WithAITools( + AIFunctionFactory.Create(GetWeather, name: "get_weather"), + AIFunctionFactory.Create(GetCurrentTime, name: "get_current_time") + ); + builder.AddAIAgent("poet", "You are a creative poet. Respond to all requests with beautiful poetry."); - builder.AddAIAgent("coder", "You are an expert programmer. Help users with coding questions and provide code examples."); + + builder.AddAIAgent("coder", "You are an expert programmer. Help users with coding questions and provide code examples.") + .WithAITool(AIFunctionFactory.Create(Add, name: "add")); // Register sample workflows var assistantBuilder = builder.AddAIAgent("workflow-assistant", "You are a helpful assistant in a workflow."); var reviewerBuilder = builder.AddAIAgent("workflow-reviewer", "You are a reviewer. Review and critique the previous response."); - builder.AddSequentialWorkflow( - "review-workflow", - [assistantBuilder, reviewerBuilder]) - .AddAsAIAgent(); - - if (builder.Environment.IsDevelopment()) + builder.AddWorkflow("review-workflow", (sp, key) => { - builder.AddDevUI(); - } + var agents = new List() { assistantBuilder, reviewerBuilder }.Select(ab => sp.GetRequiredKeyedService(ab.Name)); + return AgentWorkflowBuilder.BuildSequential(workflowName: key, agents: agents); + }).AddAsAIAgent(); + + builder.Services.AddOpenAIResponses(); + builder.Services.AddOpenAIConversations(); var app = builder.Build(); + app.MapOpenAIResponses(); + app.MapOpenAIConversations(); + if (builder.Environment.IsDevelopment()) { app.MapDevUI(); diff --git a/dotnet/samples/GettingStarted/DevUI/DevUI_Step01_BasicUsage/Properties/launchSettings.json b/dotnet/samples/GettingStarted/DevUI/DevUI_Step01_BasicUsage/Properties/launchSettings.json new file mode 100644 index 0000000000..fd55d5d1f0 --- /dev/null +++ b/dotnet/samples/GettingStarted/DevUI/DevUI_Step01_BasicUsage/Properties/launchSettings.json @@ -0,0 +1,13 @@ +{ + "profiles": { + "DevUI_Step01_BasicUsage": { + "commandName": "Project", + "launchUrl": "devui", + "launchBrowser": true, + "environmentVariables": { + "ASPNETCORE_ENVIRONMENT": "Development" + }, + "applicationUrl": "https://localhost:50516;http://localhost:50518" + } + } +} \ No newline at end of file diff --git a/dotnet/samples/GettingStarted/DevUI/DevUI_Step01_BasicUsage/README.md b/dotnet/samples/GettingStarted/DevUI/DevUI_Step01_BasicUsage/README.md index 2b6cc28644..0bf24dfb26 100644 --- a/dotnet/samples/GettingStarted/DevUI/DevUI_Step01_BasicUsage/README.md +++ b/dotnet/samples/GettingStarted/DevUI/DevUI_Step01_BasicUsage/README.md @@ -63,17 +63,23 @@ To add DevUI to your ASP.NET Core application: .AddAsAIAgent(); ``` -3. Add DevUI services and map the endpoint: +3. Add OpenAI services and map the endpoints for OpenAI and DevUI: ```csharp - builder.AddDevUI(); + // Register services for OpenAI responses and conversations (also required for DevUI) + builder.Services.AddOpenAIResponses(); + builder.Services.AddOpenAIConversations(); + var app = builder.Build(); - - app.MapDevUI(); - - // Add required endpoints - app.MapEntities(); + + // Map endpoints for OpenAI responses and conversations (also required for DevUI) app.MapOpenAIResponses(); app.MapOpenAIConversations(); + + if (builder.Environment.IsDevelopment()) + { + // Map DevUI endpoint to /devui + app.MapDevUI(); + } app.Run(); ``` diff --git a/dotnet/samples/GettingStarted/DevUI/README.md b/dotnet/samples/GettingStarted/DevUI/README.md index 155d3f2b9d..45b2f6f63b 100644 --- a/dotnet/samples/GettingStarted/DevUI/README.md +++ b/dotnet/samples/GettingStarted/DevUI/README.md @@ -38,19 +38,22 @@ builder.Services.AddChatClient(chatClient); // Register your agents builder.AddAIAgent("my-agent", "You are a helpful assistant."); -// Add DevUI services -builder.AddDevUI(); +// Register services for OpenAI responses and conversations (also required for DevUI) +builder.Services.AddOpenAIResponses(); +builder.Services.AddOpenAIConversations(); var app = builder.Build(); -// Map the DevUI endpoint -app.MapDevUI(); - -// Add required endpoints -app.MapEntities(); +// Map endpoints for OpenAI responses and conversations (also required for DevUI) app.MapOpenAIResponses(); app.MapOpenAIConversations(); +if (builder.Environment.IsDevelopment()) +{ + // Map DevUI endpoint to /devui + app.MapDevUI(); +} + app.Run(); ``` diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step01.1_Basics/FoundryAgents_Step01.1_Basics.csproj b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step01.1_Basics/FoundryAgents_Step01.1_Basics.csproj new file mode 100644 index 0000000000..89b9d8ddc0 --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step01.1_Basics/FoundryAgents_Step01.1_Basics.csproj @@ -0,0 +1,21 @@ + + + + Exe + net10.0 + + enable + enable + $(NoWarn);IDE0059 + + + + + + + + + + + + diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step01.1_Basics/Program.cs b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step01.1_Basics/Program.cs new file mode 100644 index 0000000000..9a7ee0736a --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step01.1_Basics/Program.cs @@ -0,0 +1,48 @@ +// Copyright (c) Microsoft. All rights reserved. + +// This sample shows how to create and use AI agents with Azure Foundry Agents as the backend. + +using Azure.AI.Projects; +using Azure.AI.Projects.OpenAI; +using Azure.Identity; +using Microsoft.Agents.AI; +using Microsoft.Extensions.AI; + +string endpoint = Environment.GetEnvironmentVariable("AZURE_FOUNDRY_PROJECT_ENDPOINT") ?? throw new InvalidOperationException("AZURE_FOUNDRY_PROJECT_ENDPOINT is not set."); +string deploymentName = Environment.GetEnvironmentVariable("AZURE_FOUNDRY_PROJECT_DEPLOYMENT_NAME") ?? "gpt-4o-mini"; + +const string JokerName = "JokerAgent"; + +// Get a client to create/retrieve/delete server side agents with Azure Foundry Agents. +AIProjectClient aiProjectClient = new(new Uri(endpoint), new AzureCliCredential()); + +// Define the agent you want to create. (Prompt Agent in this case) +AgentVersionCreationOptions options = new(new PromptAgentDefinition(model: deploymentName) { Instructions = "You are good at telling jokes." }); + +// Azure.AI.Agents SDK creates and manages agent by name and versions. +// You can create a server side agent version with the Azure.AI.Agents SDK client below. +AgentVersion createdAgentVersion = aiProjectClient.Agents.CreateAgentVersion(agentName: JokerName, options); + +// Note: +// agentVersion.Id = ":", +// agentVersion.Version = , +// agentVersion.Name = + +// You can retrieve an AIAgent for an already created server side agent version. +AIAgent existingJokerAgent = aiProjectClient.GetAIAgent(createdAgentVersion); + +// You can also create another AIAgent version by providing the same name with a different definition/instruction. +AIAgent newJokerAgent = aiProjectClient.CreateAIAgent(name: JokerName, model: deploymentName, instructions: "You are extremely hilarious at telling jokes."); + +// You can also get the AIAgent latest version by just providing its name. +AIAgent jokerAgentLatest = aiProjectClient.GetAIAgent(name: JokerName); +AgentVersion latestAgentVersion = jokerAgentLatest.GetService()!; + +// The AIAgent version can be accessed via the GetService method. +Console.WriteLine($"Latest agent version id: {latestAgentVersion.Id}"); + +// Once you have the AIAgent, you can invoke it like any other AIAgent. +Console.WriteLine(await jokerAgentLatest.RunAsync("Tell me a joke about a pirate.")); + +// Cleanup by agent name removes both agent versions created. +await aiProjectClient.Agents.DeleteAgentAsync(existingJokerAgent.Name); diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step01.1_Basics/README.md b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step01.1_Basics/README.md new file mode 100644 index 0000000000..ce56e05755 --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step01.1_Basics/README.md @@ -0,0 +1,40 @@ +# Creating and Managing AI Agents with Versioning + +This sample demonstrates how to create and manage AI agents with Azure Foundry Agents, including: +- Creating agents with different versions +- Retrieving agents by version or latest version +- Running multi-turn conversations with agents +- Managing agent lifecycle (creation and deletion) + +## Prerequisites + +Before you begin, ensure you have the following prerequisites: + +- .NET 10 SDK or later +- Azure Foundry service endpoint and deployment configured +- Azure CLI installed and authenticated (for Azure credential authentication) + +**Note**: This demo uses Azure CLI credentials for authentication. Make sure you're logged in with `az login` and have access to the Azure Foundry resource. For more information, see the [Azure CLI documentation](https://learn.microsoft.com/cli/azure/authenticate-azure-cli-interactively). + +Set the following environment variables: + +```powershell +$env:AZURE_FOUNDRY_PROJECT_ENDPOINT="https://your-foundry-service.services.ai.azure.com/api/projects/your-foundry-project" # Replace with your Azure Foundry resource endpoint +$env:AZURE_FOUNDRY_PROJECT_DEPLOYMENT_NAME="gpt-4o-mini" # Optional, defaults to gpt-4o-mini +``` + +## Run the sample + +Navigate to the FoundryAgents sample directory and run: + +```powershell +cd dotnet/samples/GettingStarted/FoundryAgents +dotnet run --project .\FoundryAgents_Step01.1_Basics +``` + +## What this sample demonstrates + +1. **Creating agents with versions**: Shows how to create multiple versions of the same agent with different instructions +2. **Retrieving agents**: Demonstrates retrieving agents by specific version or getting the latest version +3. **Multi-turn conversations**: Shows how to use threads to maintain conversation context across multiple agent runs +4. **Agent cleanup**: Demonstrates proper resource cleanup by deleting agents diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step01.2_Running/FoundryAgents_Step01.2_Running.csproj b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step01.2_Running/FoundryAgents_Step01.2_Running.csproj new file mode 100644 index 0000000000..daf7e24494 --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step01.2_Running/FoundryAgents_Step01.2_Running.csproj @@ -0,0 +1,20 @@ + + + + Exe + net10.0 + + enable + enable + + + + + + + + + + + + diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step01.2_Running/Program.cs b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step01.2_Running/Program.cs new file mode 100644 index 0000000000..4d840d54ff --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step01.2_Running/Program.cs @@ -0,0 +1,36 @@ +// Copyright (c) Microsoft. All rights reserved. + +// This sample shows how to create and use a simple AI agent with Azure Foundry Agents as the backend. + +using Azure.AI.Projects; +using Azure.AI.Projects.OpenAI; +using Azure.Identity; +using Microsoft.Agents.AI; + +string endpoint = Environment.GetEnvironmentVariable("AZURE_FOUNDRY_PROJECT_ENDPOINT") ?? throw new InvalidOperationException("AZURE_FOUNDRY_PROJECT_ENDPOINT is not set."); +string deploymentName = Environment.GetEnvironmentVariable("AZURE_FOUNDRY_PROJECT_DEPLOYMENT_NAME") ?? "gpt-4o-mini"; + +const string JokerInstructions = "You are good at telling jokes."; +const string JokerName = "JokerAgent"; + +// Get a client to create/retrieve/delete server side agents with Azure Foundry Agents. +AIProjectClient aiProjectClient = new(new Uri(endpoint), new AzureCliCredential()); + +// Define the agent you want to create. (Prompt Agent in this case) +AgentVersionCreationOptions options = new(new PromptAgentDefinition(model: deploymentName) { Instructions = JokerInstructions }); + +// Azure.AI.Agents SDK creates and manages agent by name and versions. +// You can create a server side agent version with the Azure.AI.Agents SDK client below. +AgentVersion agentVersion = aiProjectClient.Agents.CreateAgentVersion(agentName: JokerName, options); + +// You can retrieve an AIAgent for a already created server side agent version. +AIAgent jokerAgent = aiProjectClient.GetAIAgent(agentVersion); + +// Invoke the agent with streaming support. +await foreach (AgentRunResponseUpdate update in jokerAgent.RunStreamingAsync("Tell me a joke about a pirate.")) +{ + Console.WriteLine(update); +} + +// Cleanup by agent name removes the agent version created. +await aiProjectClient.Agents.DeleteAgentAsync(jokerAgent.Name); diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step01.2_Running/README.md b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step01.2_Running/README.md new file mode 100644 index 0000000000..53254e1975 --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step01.2_Running/README.md @@ -0,0 +1,46 @@ +# Running a Simple AI Agent with Streaming + +This sample demonstrates how to create and run a simple AI agent with Azure Foundry Agents, including both text and streaming responses. + +## What this sample demonstrates + +- Creating a simple AI agent with instructions +- Running an agent with text output +- Running an agent with streaming output +- Managing agent lifecycle (creation and deletion) + +## Prerequisites + +Before you begin, ensure you have the following prerequisites: + +- .NET 10 SDK or later +- Azure Foundry service endpoint and deployment configured +- Azure CLI installed and authenticated (for Azure credential authentication) + +**Note**: This demo uses Azure CLI credentials for authentication. Make sure you're logged in with `az login` and have access to the Azure Foundry resource. For more information, see the [Azure CLI documentation](https://learn.microsoft.com/cli/azure/authenticate-azure-cli-interactively). + +Set the following environment variables: + +```powershell +$env:AZURE_FOUNDRY_PROJECT_ENDPOINT="https://your-foundry-service.services.ai.azure.com/api/projects/your-foundry-project" # Replace with your Azure Foundry resource endpoint +$env:AZURE_FOUNDRY_PROJECT_DEPLOYMENT_NAME="gpt-4o-mini" # Optional, defaults to gpt-4o-mini +``` + +## Run the sample + +Navigate to the FoundryAgents sample directory and run: + +```powershell +cd dotnet/samples/GettingStarted/FoundryAgents +dotnet run --project .\FoundryAgents_Step01.2_Running +``` + +## Expected behavior + +The sample will: + +1. Create an agent named "JokerAgent" with instructions to tell jokes +2. Run the agent with a text prompt and display the response +3. Run the agent again with streaming to display the response as it's generated +4. Clean up resources by deleting the agent + diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step02_MultiturnConversation/FoundryAgents_Step02_MultiturnConversation.csproj b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step02_MultiturnConversation/FoundryAgents_Step02_MultiturnConversation.csproj new file mode 100644 index 0000000000..daf7e24494 --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step02_MultiturnConversation/FoundryAgents_Step02_MultiturnConversation.csproj @@ -0,0 +1,20 @@ + + + + Exe + net10.0 + + enable + enable + + + + + + + + + + + + diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step02_MultiturnConversation/Program.cs b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step02_MultiturnConversation/Program.cs new file mode 100644 index 0000000000..3cbb0099ea --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step02_MultiturnConversation/Program.cs @@ -0,0 +1,45 @@ +// Copyright (c) Microsoft. All rights reserved. + +// This sample shows how to create and use a simple AI agent with a multi-turn conversation. + +using Azure.AI.Projects; +using Azure.AI.Projects.OpenAI; +using Azure.Identity; +using Microsoft.Agents.AI; + +string endpoint = Environment.GetEnvironmentVariable("AZURE_FOUNDRY_PROJECT_ENDPOINT") ?? throw new InvalidOperationException("AZURE_FOUNDRY_PROJECT_ENDPOINT is not set."); +string deploymentName = Environment.GetEnvironmentVariable("AZURE_FOUNDRY_PROJECT_DEPLOYMENT_NAME") ?? "gpt-4o-mini"; + +const string JokerInstructions = "You are good at telling jokes."; +const string JokerName = "JokerAgent"; + +// Get a client to create/retrieve/delete server side agents with Azure Foundry Agents. +AIProjectClient aiProjectClient = new(new Uri(endpoint), new AzureCliCredential()); + +// Define the agent you want to create. (Prompt Agent in this case) +AgentVersionCreationOptions options = new(new PromptAgentDefinition(model: deploymentName) { Instructions = JokerInstructions }); + +// Create a server side agent version with the Azure.AI.Agents SDK client. +AgentVersion agentVersion = aiProjectClient.Agents.CreateAgentVersion(agentName: JokerName, options); + +// Retrieve an AIAgent for the created server side agent version. +AIAgent jokerAgent = aiProjectClient.GetAIAgent(agentVersion); + +// Invoke the agent with a multi-turn conversation, where the context is preserved in the thread object. +AgentThread thread = jokerAgent.GetNewThread(); +Console.WriteLine(await jokerAgent.RunAsync("Tell me a joke about a pirate.", thread)); +Console.WriteLine(await jokerAgent.RunAsync("Now add some emojis to the joke and tell it in the voice of a pirate's parrot.", thread)); + +// Invoke the agent with a multi-turn conversation and streaming, where the context is preserved in the thread object. +thread = jokerAgent.GetNewThread(); +await foreach (AgentRunResponseUpdate update in jokerAgent.RunStreamingAsync("Tell me a joke about a pirate.", thread)) +{ + Console.WriteLine(update); +} +await foreach (AgentRunResponseUpdate update in jokerAgent.RunStreamingAsync("Now add some emojis to the joke and tell it in the voice of a pirate's parrot.", thread)) +{ + Console.WriteLine(update); +} + +// Cleanup by agent name removes the agent version created. +await aiProjectClient.Agents.DeleteAgentAsync(jokerAgent.Name); diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step02_MultiturnConversation/README.md b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step02_MultiturnConversation/README.md new file mode 100644 index 0000000000..dab9f596db --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step02_MultiturnConversation/README.md @@ -0,0 +1,50 @@ +# Multi-turn Conversation with AI Agents + +This sample demonstrates how to implement multi-turn conversations with AI agents, where context is preserved across multiple agent runs using threads. + +## What this sample demonstrates + +- Creating an AI agent with instructions +- Using threads to maintain conversation context +- Running multi-turn conversations with text output +- Running multi-turn conversations with streaming output +- Managing agent lifecycle (creation and deletion) + +## Prerequisites + +Before you begin, ensure you have the following prerequisites: + +- .NET 10 SDK or later +- Azure Foundry service endpoint and deployment configured +- Azure CLI installed and authenticated (for Azure credential authentication) + +**Note**: This demo uses Azure CLI credentials for authentication. Make sure you're logged in with `az login` and have access to the Azure Foundry resource. For more information, see the [Azure CLI documentation](https://learn.microsoft.com/cli/azure/authenticate-azure-cli-interactively). + +Set the following environment variables: + +```powershell +$env:AZURE_FOUNDRY_PROJECT_ENDPOINT="https://your-foundry-service.services.ai.azure.com/api/projects/your-foundry-project" # Replace with your Azure Foundry resource endpoint +$env:AZURE_FOUNDRY_PROJECT_DEPLOYMENT_NAME="gpt-4o-mini" # Optional, defaults to gpt-4o-mini +``` + +## Run the sample + +Navigate to the FoundryAgents sample directory and run: + +```powershell +cd dotnet/samples/GettingStarted/FoundryAgents +dotnet run --project .\FoundryAgents_Step02_MultiturnConversation +``` + +## Expected behavior + +The sample will: + +1. Create an agent named "JokerAgent" with instructions to tell jokes +2. Create a thread for conversation context +3. Run the agent with a text prompt and display the response +4. Send a follow-up message to the same thread, demonstrating context preservation +5. Create a new thread and run the agent with streaming +6. Send a follow-up streaming message to demonstrate multi-turn streaming +7. Clean up resources by deleting the agent + diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step03_UsingFunctionTools/FoundryAgents_Step03_UsingFunctionTools.csproj b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step03_UsingFunctionTools/FoundryAgents_Step03_UsingFunctionTools.csproj new file mode 100644 index 0000000000..daf7e24494 --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step03_UsingFunctionTools/FoundryAgents_Step03_UsingFunctionTools.csproj @@ -0,0 +1,20 @@ + + + + Exe + net10.0 + + enable + enable + + + + + + + + + + + + diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step03_UsingFunctionTools/Program.cs b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step03_UsingFunctionTools/Program.cs new file mode 100644 index 0000000000..38c5a15d75 --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step03_UsingFunctionTools/Program.cs @@ -0,0 +1,51 @@ +// Copyright (c) Microsoft. All rights reserved. + +// This sample demonstrates how to use an agent with function tools. +// It shows both non-streaming and streaming agent interactions using weather-related tools. + +using System.ComponentModel; +using Azure.AI.Projects; +using Azure.Identity; +using Microsoft.Agents.AI; +using Microsoft.Extensions.AI; + +string endpoint = Environment.GetEnvironmentVariable("AZURE_FOUNDRY_PROJECT_ENDPOINT") ?? throw new InvalidOperationException("AZURE_FOUNDRY_PROJECT_ENDPOINT is not set."); +string deploymentName = Environment.GetEnvironmentVariable("AZURE_FOUNDRY_PROJECT_DEPLOYMENT_NAME") ?? "gpt-4o-mini"; + +[Description("Get the weather for a given location.")] +static string GetWeather([Description("The location to get the weather for.")] string location) + => $"The weather in {location} is cloudy with a high of 15°C."; + +const string AssistantInstructions = "You are a helpful assistant that can get weather information."; +const string AssistantName = "WeatherAssistant"; + +// Get a client to create/retrieve/delete server side agents with Azure Foundry Agents. +AIProjectClient aiProjectClient = new(new Uri(endpoint), new AzureCliCredential()); + +// Define the agent with function tools. +AITool tool = AIFunctionFactory.Create(GetWeather); + +// Create AIAgent directly +var newAgent = await aiProjectClient.CreateAIAgentAsync(name: AssistantName, model: deploymentName, instructions: AssistantInstructions, tools: [tool]); + +// Getting an already existing agent by name with tools. +/* + * IMPORTANT: Since agents that are stored in the server only know the definition of the function tools (JSON Schema), + * you need to provided all invocable function tools when retrieving the agent so it can invoke them automatically. + * If no invocable tools are provided, the function calling needs to handled manually. + */ +var existingAgent = await aiProjectClient.GetAIAgentAsync(name: AssistantName, tools: [tool]); + +// Non-streaming agent interaction with function tools. +AgentThread thread = existingAgent.GetNewThread(); +Console.WriteLine(await existingAgent.RunAsync("What is the weather like in Amsterdam?", thread)); + +// Streaming agent interaction with function tools. +thread = existingAgent.GetNewThread(); +await foreach (AgentRunResponseUpdate update in existingAgent.RunStreamingAsync("What is the weather like in Amsterdam?", thread)) +{ + Console.WriteLine(update); +} + +// Cleanup by agent name removes the agent version created. +await aiProjectClient.Agents.DeleteAgentAsync(existingAgent.Name); diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step03_UsingFunctionTools/README.md b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step03_UsingFunctionTools/README.md new file mode 100644 index 0000000000..35bef8a999 --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step03_UsingFunctionTools/README.md @@ -0,0 +1,48 @@ +# Using Function Tools with AI Agents + +This sample demonstrates how to use function tools with AI agents, allowing agents to call custom functions to retrieve information. + +## What this sample demonstrates + +- Creating function tools using AIFunctionFactory +- Passing function tools to an AI agent +- Running agents with function tools (text output) +- Running agents with function tools (streaming output) +- Managing agent lifecycle (creation and deletion) + +## Prerequisites + +Before you begin, ensure you have the following prerequisites: + +- .NET 10 SDK or later +- Azure Foundry service endpoint and deployment configured +- Azure CLI installed and authenticated (for Azure credential authentication) + +**Note**: This demo uses Azure CLI credentials for authentication. Make sure you're logged in with `az login` and have access to the Azure Foundry resource. For more information, see the [Azure CLI documentation](https://learn.microsoft.com/cli/azure/authenticate-azure-cli-interactively). + +Set the following environment variables: + +```powershell +$env:AZURE_FOUNDRY_PROJECT_ENDPOINT="https://your-foundry-service.services.ai.azure.com/api/projects/your-foundry-project" # Replace with your Azure Foundry resource endpoint +$env:AZURE_FOUNDRY_PROJECT_DEPLOYMENT_NAME="gpt-4o-mini" # Optional, defaults to gpt-4o-mini +``` + +## Run the sample + +Navigate to the FoundryAgents sample directory and run: + +```powershell +cd dotnet/samples/GettingStarted/FoundryAgents +dotnet run --project .\FoundryAgents_Step03.1_UsingFunctionTools +``` + +## Expected behavior + +The sample will: + +1. Create an agent named "WeatherAssistant" with a GetWeather function tool +2. Run the agent with a text prompt asking about weather +3. The agent will invoke the GetWeather function tool to retrieve weather information +4. Run the agent again with streaming to display the response as it's generated +5. Clean up resources by deleting the agent + diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step04_UsingFunctionToolsWithApprovals/FoundryAgents_Step04_UsingFunctionToolsWithApprovals.csproj b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step04_UsingFunctionToolsWithApprovals/FoundryAgents_Step04_UsingFunctionToolsWithApprovals.csproj new file mode 100644 index 0000000000..daf7e24494 --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step04_UsingFunctionToolsWithApprovals/FoundryAgents_Step04_UsingFunctionToolsWithApprovals.csproj @@ -0,0 +1,20 @@ + + + + Exe + net10.0 + + enable + enable + + + + + + + + + + + + diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step04_UsingFunctionToolsWithApprovals/Program.cs b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step04_UsingFunctionToolsWithApprovals/Program.cs new file mode 100644 index 0000000000..1b51d210cf --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step04_UsingFunctionToolsWithApprovals/Program.cs @@ -0,0 +1,64 @@ +// Copyright (c) Microsoft. All rights reserved. + +// This sample demonstrates how to use an agent with function tools that require a human in the loop for approvals. +// It shows both non-streaming and streaming agent interactions using weather-related tools. +// If the agent is hosted in a service, with a remote user, combine this sample with the Persisted Conversations sample to persist the chat history +// while the agent is waiting for user input. + +using System.ComponentModel; +using Azure.AI.Projects; +using Azure.Identity; +using Microsoft.Agents.AI; +using Microsoft.Extensions.AI; + +string endpoint = Environment.GetEnvironmentVariable("AZURE_FOUNDRY_PROJECT_ENDPOINT") ?? throw new InvalidOperationException("AZURE_FOUNDRY_PROJECT_ENDPOINT is not set."); +string deploymentName = Environment.GetEnvironmentVariable("AZURE_FOUNDRY_PROJECT_DEPLOYMENT_NAME") ?? "gpt-4o-mini"; + +// Create a sample function tool that the agent can use. +[Description("Get the weather for a given location.")] +static string GetWeather([Description("The location to get the weather for.")] string location) + => $"The weather in {location} is cloudy with a high of 15°C."; + +const string AssistantInstructions = "You are a helpful assistant that can get weather information."; +const string AssistantName = "WeatherAssistant"; + +// Get a client to create/retrieve/delete server side agents with Azure Foundry Agents. +AIProjectClient aiProjectClient = new(new Uri(endpoint), new AzureCliCredential()); + +ApprovalRequiredAIFunction approvalTool = new(AIFunctionFactory.Create(GetWeather, name: nameof(GetWeather))); + +// Create AIAgent directly +AIAgent agent = await aiProjectClient.CreateAIAgentAsync(name: AssistantName, model: deploymentName, instructions: AssistantInstructions, tools: [approvalTool]); + +// Call the agent with approval-required function tools. +// The agent will request approval before invoking the function. +AgentThread thread = agent.GetNewThread(); +AgentRunResponse response = await agent.RunAsync("What is the weather like in Amsterdam?", thread); + +// Check if there are any user input requests (approvals needed). +List userInputRequests = response.UserInputRequests.ToList(); + +while (userInputRequests.Count > 0) +{ + // Ask the user to approve each function call request. + // For simplicity, we are assuming here that only function approval requests are being made. + List userInputMessages = userInputRequests + .OfType() + .Select(functionApprovalRequest => + { + Console.WriteLine($"The agent would like to invoke the following function, please reply Y to approve: Name {functionApprovalRequest.FunctionCall.Name}"); + bool approved = Console.ReadLine()?.Equals("Y", StringComparison.OrdinalIgnoreCase) ?? false; + return new ChatMessage(ChatRole.User, [functionApprovalRequest.CreateResponse(approved)]); + }) + .ToList(); + + // Pass the user input responses back to the agent for further processing. + response = await agent.RunAsync(userInputMessages, thread); + + userInputRequests = response.UserInputRequests.ToList(); +} + +Console.WriteLine($"\nAgent: {response}"); + +// Cleanup by agent name removes the agent version created. +await aiProjectClient.Agents.DeleteAgentAsync(agent.Name); diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step04_UsingFunctionToolsWithApprovals/README.md b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step04_UsingFunctionToolsWithApprovals/README.md new file mode 100644 index 0000000000..5a797acd0f --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step04_UsingFunctionToolsWithApprovals/README.md @@ -0,0 +1,51 @@ +# Using Function Tools with Approvals (Human-in-the-Loop) + +This sample demonstrates how to use function tools that require human approval before execution, implementing a human-in-the-loop workflow. + +## What this sample demonstrates + +- Creating approval-required function tools using ApprovalRequiredAIFunction +- Handling user input requests for function approvals +- Implementing human-in-the-loop approval workflows +- Processing agent responses with pending approvals +- Managing agent lifecycle (creation and deletion) + +## Prerequisites + +Before you begin, ensure you have the following prerequisites: + +- .NET 10 SDK or later +- Azure Foundry service endpoint and deployment configured +- Azure CLI installed and authenticated (for Azure credential authentication) + +**Note**: This demo uses Azure CLI credentials for authentication. Make sure you're logged in with `az login` and have access to the Azure Foundry resource. For more information, see the [Azure CLI documentation](https://learn.microsoft.com/cli/azure/authenticate-azure-cli-interactively). + +Set the following environment variables: + +```powershell +$env:AZURE_FOUNDRY_PROJECT_ENDPOINT="https://your-foundry-service.services.ai.azure.com/api/projects/your-foundry-project" # Replace with your Azure Foundry resource endpoint +$env:AZURE_FOUNDRY_PROJECT_DEPLOYMENT_NAME="gpt-4o-mini" # Optional, defaults to gpt-4o-mini +``` + +## Run the sample + +Navigate to the FoundryAgents sample directory and run: + +```powershell +cd dotnet/samples/GettingStarted/FoundryAgents +dotnet run --project .\FoundryAgents_Step04_UsingFunctionToolsWithApprovals +``` + +## Expected behavior + +The sample will: + +1. Create an agent named "WeatherAssistant" with an approval-required GetWeather function tool +2. Run the agent with a prompt asking about weather +3. The agent will request approval before invoking the GetWeather function +4. The sample will prompt the user to approve or deny the function call (enter 'Y' to approve) +5. After approval, the function will be executed and the result returned to the agent +6. Clean up resources by deleting the agent + +**Note**: For hosted agents with remote users, combine this sample with the Persisted Conversations sample to persist chat history while waiting for user approval. + diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step05_StructuredOutput/FoundryAgents_Step05_StructuredOutput.csproj b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step05_StructuredOutput/FoundryAgents_Step05_StructuredOutput.csproj new file mode 100644 index 0000000000..daf7e24494 --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step05_StructuredOutput/FoundryAgents_Step05_StructuredOutput.csproj @@ -0,0 +1,20 @@ + + + + Exe + net10.0 + + enable + enable + + + + + + + + + + + + diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step05_StructuredOutput/Program.cs b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step05_StructuredOutput/Program.cs new file mode 100644 index 0000000000..ac05565836 --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step05_StructuredOutput/Program.cs @@ -0,0 +1,91 @@ +// Copyright (c) Microsoft. All rights reserved. + +// This sample shows how to configure an agent to produce structured output. + +using System.ComponentModel; +using System.Text.Json; +using System.Text.Json.Serialization; +using Azure.AI.Projects; +using Azure.Identity; +using Microsoft.Agents.AI; +using SampleApp; + +#pragma warning disable CA5399 + +string endpoint = Environment.GetEnvironmentVariable("AZURE_FOUNDRY_PROJECT_ENDPOINT") ?? throw new InvalidOperationException("AZURE_FOUNDRY_PROJECT_ENDPOINT is not set."); +string deploymentName = Environment.GetEnvironmentVariable("AZURE_FOUNDRY_PROJECT_DEPLOYMENT_NAME") ?? "gpt-4o-mini"; + +const string AssistantInstructions = "You are a helpful assistant that extracts structured information about people."; +const string AssistantName = "StructuredOutputAssistant"; + +// Get a client to create/retrieve/delete server side agents with Azure Foundry Agents. +AIProjectClient aiProjectClient = new(new Uri(endpoint), new AzureCliCredential()); + +// Create ChatClientAgent directly +ChatClientAgent agent = await aiProjectClient.CreateAIAgentAsync( + model: deploymentName, + new ChatClientAgentOptions() + { + Name = AssistantName, + ChatOptions = new() + { + Instructions = AssistantInstructions, + ResponseFormat = Microsoft.Extensions.AI.ChatResponseFormat.ForJsonSchema() + } + }); + +// Set PersonInfo as the type parameter of RunAsync method to specify the expected structured output from the agent and invoke the agent with some unstructured input. +AgentRunResponse response = await agent.RunAsync("Please provide information about John Smith, who is a 35-year-old software engineer."); + +// Access the structured output via the Result property of the agent response. +Console.WriteLine("Assistant Output:"); +Console.WriteLine($"Name: {response.Result.Name}"); +Console.WriteLine($"Age: {response.Result.Age}"); +Console.WriteLine($"Occupation: {response.Result.Occupation}"); + +// Create the ChatClientAgent with the specified name, instructions, and expected structured output the agent should produce. +ChatClientAgent agentWithPersonInfo = aiProjectClient.CreateAIAgent( + model: deploymentName, + new ChatClientAgentOptions() + { + Name = AssistantName, + ChatOptions = new() + { + Instructions = AssistantInstructions, + ResponseFormat = Microsoft.Extensions.AI.ChatResponseFormat.ForJsonSchema() + } + }); + +// Invoke the agent with some unstructured input while streaming, to extract the structured information from. +IAsyncEnumerable updates = agentWithPersonInfo.RunStreamingAsync("Please provide information about John Smith, who is a 35-year-old software engineer."); + +// Assemble all the parts of the streamed output, since we can only deserialize once we have the full json, +// then deserialize the response into the PersonInfo class. +PersonInfo personInfo = (await updates.ToAgentRunResponseAsync()).Deserialize(JsonSerializerOptions.Web); + +Console.WriteLine("Assistant Output:"); +Console.WriteLine($"Name: {personInfo.Name}"); +Console.WriteLine($"Age: {personInfo.Age}"); +Console.WriteLine($"Occupation: {personInfo.Occupation}"); + +// Cleanup by agent name removes the agent version created. +await aiProjectClient.Agents.DeleteAgentAsync(agent.Name); + +namespace SampleApp +{ + /// + /// Represents information about a person, including their name, age, and occupation, matched to the JSON schema used in the agent. + /// + [Description("Information about a person including their name, age, and occupation")] + public class PersonInfo + { + [JsonPropertyName("name")] + public string? Name { get; set; } + + [JsonPropertyName("age")] + public int? Age { get; set; } + + [JsonPropertyName("occupation")] + public string? Occupation { get; set; } + } +} diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step05_StructuredOutput/README.md b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step05_StructuredOutput/README.md new file mode 100644 index 0000000000..956a2542e9 --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step05_StructuredOutput/README.md @@ -0,0 +1,49 @@ +# Structured Output with AI Agents + +This sample demonstrates how to configure AI agents to produce structured output in JSON format using JSON schemas. + +## What this sample demonstrates + +- Configuring agents with JSON schema response formats +- Using generic RunAsync method for structured output +- Deserializing structured responses into typed objects +- Running agents with streaming and structured output +- Managing agent lifecycle (creation and deletion) + +## Prerequisites + +Before you begin, ensure you have the following prerequisites: + +- .NET 10 SDK or later +- Azure Foundry service endpoint and deployment configured +- Azure CLI installed and authenticated (for Azure credential authentication) + +**Note**: This demo uses Azure CLI credentials for authentication. Make sure you're logged in with `az login` and have access to the Azure Foundry resource. For more information, see the [Azure CLI documentation](https://learn.microsoft.com/cli/azure/authenticate-azure-cli-interactively). + +Set the following environment variables: + +```powershell +$env:AZURE_FOUNDRY_PROJECT_ENDPOINT="https://your-foundry-service.services.ai.azure.com/api/projects/your-foundry-project" # Replace with your Azure Foundry resource endpoint +$env:AZURE_FOUNDRY_PROJECT_DEPLOYMENT_NAME="gpt-4o-mini" # Optional, defaults to gpt-4o-mini +``` + +## Run the sample + +Navigate to the FoundryAgents sample directory and run: + +```powershell +cd dotnet/samples/GettingStarted/FoundryAgents +dotnet run --project .\FoundryAgents_Step05_StructuredOutput +``` + +## Expected behavior + +The sample will: + +1. Create an agent named "StructuredOutputAssistant" configured to produce JSON output +2. Run the agent with a prompt to extract person information +3. Deserialize the JSON response into a PersonInfo object +4. Display the structured data (Name, Age, Occupation) +5. Run the agent again with streaming and deserialize the streamed JSON response +6. Clean up resources by deleting the agent + diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step06_PersistedConversations/FoundryAgents_Step06_PersistedConversations.csproj b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step06_PersistedConversations/FoundryAgents_Step06_PersistedConversations.csproj new file mode 100644 index 0000000000..daf7e24494 --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step06_PersistedConversations/FoundryAgents_Step06_PersistedConversations.csproj @@ -0,0 +1,20 @@ + + + + Exe + net10.0 + + enable + enable + + + + + + + + + + + + diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step06_PersistedConversations/Program.cs b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step06_PersistedConversations/Program.cs new file mode 100644 index 0000000000..d404a814c0 --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step06_PersistedConversations/Program.cs @@ -0,0 +1,44 @@ +// Copyright (c) Microsoft. All rights reserved. + +// This sample shows how to create and use a simple AI agent with a conversation that can be persisted to disk. + +using System.Text.Json; +using Azure.AI.Projects; +using Azure.Identity; +using Microsoft.Agents.AI; + +string endpoint = Environment.GetEnvironmentVariable("AZURE_FOUNDRY_PROJECT_ENDPOINT") ?? throw new InvalidOperationException("AZURE_FOUNDRY_PROJECT_ENDPOINT is not set."); +string deploymentName = Environment.GetEnvironmentVariable("AZURE_FOUNDRY_PROJECT_DEPLOYMENT_NAME") ?? "gpt-4o-mini"; + +const string JokerInstructions = "You are good at telling jokes."; +const string JokerName = "JokerAgent"; + +// Get a client to create/retrieve/delete server side agents with Azure Foundry Agents. +AIProjectClient aiProjectClient = new(new Uri(endpoint), new AzureCliCredential()); + +AIAgent agent = await aiProjectClient.CreateAIAgentAsync(name: JokerName, model: deploymentName, instructions: JokerInstructions); + +// Start a new thread for the agent conversation. +AgentThread thread = agent.GetNewThread(); + +// Run the agent with a new thread. +Console.WriteLine(await agent.RunAsync("Tell me a joke about a pirate.", thread)); + +// Serialize the thread state to a JsonElement, so it can be stored for later use. +JsonElement serializedThread = thread.Serialize(); + +// Save the serialized thread to a temporary file (for demonstration purposes). +string tempFilePath = Path.GetTempFileName(); +await File.WriteAllTextAsync(tempFilePath, JsonSerializer.Serialize(serializedThread)); + +// Load the serialized thread from the temporary file (for demonstration purposes). +JsonElement reloadedSerializedThread = JsonElement.Parse(await File.ReadAllTextAsync(tempFilePath))!; + +// Deserialize the thread state after loading from storage. +AgentThread resumedThread = agent.DeserializeThread(reloadedSerializedThread); + +// Run the agent again with the resumed thread. +Console.WriteLine(await agent.RunAsync("Now tell the same joke in the voice of a pirate, and add some emojis to the joke.", resumedThread)); + +// Cleanup by agent name removes the agent version created. +await aiProjectClient.Agents.DeleteAgentAsync(agent.Name); diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step06_PersistedConversations/README.md b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step06_PersistedConversations/README.md new file mode 100644 index 0000000000..29c2233748 --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step06_PersistedConversations/README.md @@ -0,0 +1,50 @@ +# Persisted Conversations with AI Agents + +This sample demonstrates how to serialize and persist agent conversation threads to storage, allowing conversations to be resumed later. + +## What this sample demonstrates + +- Serializing agent threads to JSON +- Persisting thread state to disk +- Loading and deserializing thread state from storage +- Resuming conversations with persisted threads +- Managing agent lifecycle (creation and deletion) + +## Prerequisites + +Before you begin, ensure you have the following prerequisites: + +- .NET 10 SDK or later +- Azure Foundry service endpoint and deployment configured +- Azure CLI installed and authenticated (for Azure credential authentication) + +**Note**: This demo uses Azure CLI credentials for authentication. Make sure you're logged in with `az login` and have access to the Azure Foundry resource. For more information, see the [Azure CLI documentation](https://learn.microsoft.com/cli/azure/authenticate-azure-cli-interactively). + +Set the following environment variables: + +```powershell +$env:AZURE_FOUNDRY_PROJECT_ENDPOINT="https://your-foundry-service.services.ai.azure.com/api/projects/your-foundry-project" # Replace with your Azure Foundry resource endpoint +$env:AZURE_FOUNDRY_PROJECT_DEPLOYMENT_NAME="gpt-4o-mini" # Optional, defaults to gpt-4o-mini +``` + +## Run the sample + +Navigate to the FoundryAgents sample directory and run: + +```powershell +cd dotnet/samples/GettingStarted/FoundryAgents +dotnet run --project .\FoundryAgents_Step06_PersistedConversations +``` + +## Expected behavior + +The sample will: + +1. Create an agent named "JokerAgent" with instructions to tell jokes +2. Create a thread and run the agent with an initial prompt +3. Serialize the thread state to JSON +4. Save the serialized thread to a temporary file +5. Load the thread from the file and deserialize it +6. Resume the conversation with the same thread using a follow-up prompt +7. Clean up resources by deleting the agent + diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step07_Observability/FoundryAgents_Step07_Observability.csproj b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step07_Observability/FoundryAgents_Step07_Observability.csproj new file mode 100644 index 0000000000..5ceeabb204 --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step07_Observability/FoundryAgents_Step07_Observability.csproj @@ -0,0 +1,23 @@ + + + + Exe + net10.0 + + enable + enable + + + + + + + + + + + + + + + diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step07_Observability/Program.cs b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step07_Observability/Program.cs new file mode 100644 index 0000000000..eb011ba064 --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step07_Observability/Program.cs @@ -0,0 +1,52 @@ +// Copyright (c) Microsoft. All rights reserved. + +// This sample shows how to create and use a simple AI agent with Azure Foundry Agents as the backend that logs telemetry using OpenTelemetry. + +using Azure.AI.Projects; +using Azure.Identity; +using Azure.Monitor.OpenTelemetry.Exporter; +using Microsoft.Agents.AI; +using OpenTelemetry; +using OpenTelemetry.Trace; + +string endpoint = Environment.GetEnvironmentVariable("AZURE_FOUNDRY_PROJECT_ENDPOINT") ?? throw new InvalidOperationException("AZURE_FOUNDRY_PROJECT_ENDPOINT is not set."); +string deploymentName = Environment.GetEnvironmentVariable("AZURE_FOUNDRY_PROJECT_DEPLOYMENT_NAME") ?? "gpt-4o-mini"; +string? applicationInsightsConnectionString = Environment.GetEnvironmentVariable("APPLICATIONINSIGHTS_CONNECTION_STRING"); + +const string JokerInstructions = "You are good at telling jokes."; +const string JokerName = "JokerAgent"; + +// Create TracerProvider with console exporter +// This will output the telemetry data to the console. +string sourceName = Guid.NewGuid().ToString("N"); +TracerProviderBuilder tracerProviderBuilder = Sdk.CreateTracerProviderBuilder() + .AddSource(sourceName) + .AddConsoleExporter(); +if (!string.IsNullOrWhiteSpace(applicationInsightsConnectionString)) +{ + tracerProviderBuilder.AddAzureMonitorTraceExporter(options => options.ConnectionString = applicationInsightsConnectionString); +} +using var tracerProvider = tracerProviderBuilder.Build(); + +// Get a client to create/retrieve/delete server side agents with Azure Foundry Agents. +AIProjectClient aiProjectClient = new(new Uri(endpoint), new AzureCliCredential()); + +// Define the agent you want to create. (Prompt Agent in this case) +AIAgent agent = aiProjectClient.CreateAIAgent(name: JokerName, model: deploymentName, instructions: JokerInstructions) + .AsBuilder() + .UseOpenTelemetry(sourceName: sourceName) + .Build(); + +// Invoke the agent and output the text result. +AgentThread thread = agent.GetNewThread(); +Console.WriteLine(await agent.RunAsync("Tell me a joke about a pirate.", thread)); + +// Invoke the agent with streaming support. +thread = agent.GetNewThread(); +await foreach (AgentRunResponseUpdate update in agent.RunStreamingAsync("Tell me a joke about a pirate.", thread)) +{ + Console.WriteLine(update); +} + +// Cleanup by agent name removes the agent version created. +await aiProjectClient.Agents.DeleteAgentAsync(agent.Name); diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step07_Observability/README.md b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step07_Observability/README.md new file mode 100644 index 0000000000..30f7014dff --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step07_Observability/README.md @@ -0,0 +1,51 @@ +# Observability with OpenTelemetry + +This sample demonstrates how to add observability to AI agents using OpenTelemetry for tracing and monitoring. + +## What this sample demonstrates + +- Setting up OpenTelemetry TracerProvider +- Configuring console exporter for telemetry output +- Configuring Azure Monitor exporter for Application Insights +- Adding OpenTelemetry middleware to agents +- Running agents with telemetry collection (text and streaming) +- Managing agent lifecycle (creation and deletion) + +## Prerequisites + +Before you begin, ensure you have the following prerequisites: + +- .NET 10 SDK or later +- Azure Foundry service endpoint and deployment configured +- Azure CLI installed and authenticated (for Azure credential authentication) +- (Optional) Application Insights connection string for Azure Monitor integration + +**Note**: This demo uses Azure CLI credentials for authentication. Make sure you're logged in with `az login` and have access to the Azure Foundry resource. For more information, see the [Azure CLI documentation](https://learn.microsoft.com/cli/azure/authenticate-azure-cli-interactively). + +Set the following environment variables: + +```powershell +$env:AZURE_FOUNDRY_PROJECT_ENDPOINT="https://your-foundry-service.services.ai.azure.com/api/projects/your-foundry-project" # Replace with your Azure Foundry resource endpoint +$env:AZURE_FOUNDRY_PROJECT_DEPLOYMENT_NAME="gpt-4o-mini" # Optional, defaults to gpt-4o-mini +$env:APPLICATIONINSIGHTS_CONNECTION_STRING="your-connection-string" # Optional, for Azure Monitor integration +``` + +## Run the sample + +Navigate to the FoundryAgents sample directory and run: + +```powershell +cd dotnet/samples/GettingStarted/FoundryAgents +dotnet run --project .\FoundryAgents_Step07_Observability +``` + +## Expected behavior + +The sample will: + +1. Create a TracerProvider with console exporter (and optionally Azure Monitor exporter) +2. Create an agent named "JokerAgent" with OpenTelemetry middleware +3. Run the agent with a text prompt and display telemetry traces to console +4. Run the agent again with streaming and display telemetry traces +5. Clean up resources by deleting the agent + diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step08_DependencyInjection/FoundryAgents_Step08_DependencyInjection.csproj b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step08_DependencyInjection/FoundryAgents_Step08_DependencyInjection.csproj new file mode 100644 index 0000000000..f1812befeb --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step08_DependencyInjection/FoundryAgents_Step08_DependencyInjection.csproj @@ -0,0 +1,23 @@ + + + + Exe + net10.0 + + enable + enable + + $(NoWarn);CA1812 + + + + + + + + + + + + + diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step08_DependencyInjection/Program.cs b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step08_DependencyInjection/Program.cs new file mode 100644 index 0000000000..4bf4843d66 --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step08_DependencyInjection/Program.cs @@ -0,0 +1,82 @@ +// Copyright (c) Microsoft. All rights reserved. + +// This sample shows how to use dependency injection to register an AIAgent and use it from a hosted service with a user input chat loop. + +using Azure.AI.Projects; +using Azure.Identity; +using Microsoft.Agents.AI; +using Microsoft.Extensions.DependencyInjection; +using Microsoft.Extensions.Hosting; + +string endpoint = Environment.GetEnvironmentVariable("AZURE_FOUNDRY_PROJECT_ENDPOINT") ?? throw new InvalidOperationException("AZURE_FOUNDRY_PROJECT_ENDPOINT is not set."); +string deploymentName = Environment.GetEnvironmentVariable("AZURE_FOUNDRY_PROJECT_DEPLOYMENT_NAME") ?? "gpt-4o-mini"; + +const string JokerInstructions = "You are good at telling jokes."; +const string JokerName = "JokerAgent"; + +// Create a host builder that we will register services with and then run. +HostApplicationBuilder builder = Host.CreateApplicationBuilder(args); + +// Add the agents client to the service collection. +builder.Services.AddSingleton((sp) => new AIProjectClient(new Uri(endpoint), new AzureCliCredential())); + +// Add the AI agent to the service collection. +builder.Services.AddSingleton((sp) + => sp.GetRequiredService() + .CreateAIAgent(name: JokerName, model: deploymentName, instructions: JokerInstructions)); + +// Add a sample service that will use the agent to respond to user input. +builder.Services.AddHostedService(); + +// Build and run the host. +using IHost host = builder.Build(); +await host.RunAsync().ConfigureAwait(false); + +/// +/// A sample service that uses an AI agent to respond to user input. +/// +internal sealed class SampleService(AIProjectClient client, AIAgent agent, IHostApplicationLifetime appLifetime) : IHostedService +{ + private AgentThread? _thread; + + public async Task StartAsync(CancellationToken cancellationToken) + { + // Create a thread that will be used for the entirety of the service lifetime so that the user can ask follow up questions. + this._thread = agent.GetNewThread(); + _ = this.RunAsync(appLifetime.ApplicationStopping); + } + + public async Task RunAsync(CancellationToken cancellationToken) + { + // Delay a little to allow the service to finish starting. + await Task.Delay(100, cancellationToken); + + while (!cancellationToken.IsCancellationRequested) + { + Console.WriteLine("\nAgent: Ask me to tell you a joke about a specific topic. To exit just press Ctrl+C or enter without any input.\n"); + Console.Write("> "); + string? input = Console.ReadLine(); + + // If the user enters no input, signal the application to shut down. + if (string.IsNullOrWhiteSpace(input)) + { + appLifetime.StopApplication(); + break; + } + + // Stream the output to the console as it is generated. + await foreach (AgentRunResponseUpdate update in agent.RunStreamingAsync(input, this._thread, cancellationToken: cancellationToken)) + { + Console.Write(update); + } + + Console.WriteLine(); + } + } + + public async Task StopAsync(CancellationToken cancellationToken) + { + Console.WriteLine("\nDeleting agent ..."); + await client.Agents.DeleteAgentAsync(agent.Name, cancellationToken).ConfigureAwait(false); + } +} diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step08_DependencyInjection/README.md b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step08_DependencyInjection/README.md new file mode 100644 index 0000000000..580821bb0a --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step08_DependencyInjection/README.md @@ -0,0 +1,51 @@ +# Dependency Injection with AI Agents + +This sample demonstrates how to use dependency injection to register and manage AI agents within a hosted service application. + +## What this sample demonstrates + +- Setting up dependency injection with HostApplicationBuilder +- Registering AIProjectClient as a singleton service +- Registering AIAgent as a singleton service +- Using agents in hosted services +- Interactive chat loop with streaming responses +- Managing agent lifecycle (creation and deletion) + +## Prerequisites + +Before you begin, ensure you have the following prerequisites: + +- .NET 10 SDK or later +- Azure Foundry service endpoint and deployment configured +- Azure CLI installed and authenticated (for Azure credential authentication) + +**Note**: This demo uses Azure CLI credentials for authentication. Make sure you're logged in with `az login` and have access to the Azure Foundry resource. For more information, see the [Azure CLI documentation](https://learn.microsoft.com/cli/azure/authenticate-azure-cli-interactively). + +Set the following environment variables: + +```powershell +$env:AZURE_FOUNDRY_PROJECT_ENDPOINT="https://your-foundry-service.services.ai.azure.com/api/projects/your-foundry-project" # Replace with your Azure Foundry resource endpoint +$env:AZURE_FOUNDRY_PROJECT_DEPLOYMENT_NAME="gpt-4o-mini" # Optional, defaults to gpt-4o-mini +``` + +## Run the sample + +Navigate to the FoundryAgents sample directory and run: + +```powershell +cd dotnet/samples/GettingStarted/FoundryAgents +dotnet run --project .\FoundryAgents_Step08_DependencyInjection +``` + +## Expected behavior + +The sample will: + +1. Create a host with dependency injection configured +2. Register AIProjectClient and AIAgent as services +3. Create an agent named "JokerAgent" with instructions to tell jokes +4. Start an interactive chat loop where you can ask the agent questions +5. The agent will respond with streaming output +6. Enter an empty line or press Ctrl+C to exit +7. Clean up resources by deleting the agent + diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step09_UsingMcpClientAsTools/FoundryAgents_Step09_UsingMcpClientAsTools.csproj b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step09_UsingMcpClientAsTools/FoundryAgents_Step09_UsingMcpClientAsTools.csproj new file mode 100644 index 0000000000..a6d96cb3db --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step09_UsingMcpClientAsTools/FoundryAgents_Step09_UsingMcpClientAsTools.csproj @@ -0,0 +1,23 @@ + + + + Exe + net10.0 + + enable + enable + 3afc9b74-af74-4d8e-ae96-fa1c511d11ac + + + + + + + + + + + + + + diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step09_UsingMcpClientAsTools/Program.cs b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step09_UsingMcpClientAsTools/Program.cs new file mode 100644 index 0000000000..a821c1194b --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step09_UsingMcpClientAsTools/Program.cs @@ -0,0 +1,47 @@ +// Copyright (c) Microsoft. All rights reserved. + +// This sample shows how to expose an AI agent as an MCP tool. + +using Azure.AI.Projects; +using Azure.Identity; +using Microsoft.Agents.AI; +using Microsoft.Extensions.AI; +using ModelContextProtocol.Client; + +string endpoint = Environment.GetEnvironmentVariable("AZURE_FOUNDRY_PROJECT_ENDPOINT") ?? throw new InvalidOperationException("AZURE_FOUNDRY_PROJECT_ENDPOINT is not set."); +string deploymentName = Environment.GetEnvironmentVariable("AZURE_FOUNDRY_PROJECT_DEPLOYMENT_NAME") ?? "gpt-4o-mini"; + +Console.WriteLine("Starting MCP Stdio for @modelcontextprotocol/server-github ... "); + +// Create an MCPClient for the GitHub server +await using var mcpClient = await McpClient.CreateAsync(new StdioClientTransport(new() +{ + Name = "MCPServer", + Command = "npx", + Arguments = ["-y", "--verbose", "@modelcontextprotocol/server-github"], +})); + +// Retrieve the list of tools available on the GitHub server +IList mcpTools = await mcpClient.ListToolsAsync(); +string agentName = "AgentWithMCP"; +// Get a client to create/retrieve/delete server side agents with Azure Foundry Agents. +AIProjectClient aiProjectClient = new(new Uri(endpoint), new AzureCliCredential()); + +Console.WriteLine($"Creating the agent '{agentName}' ..."); + +// Define the agent you want to create. (Prompt Agent in this case) +AIAgent agent = aiProjectClient.CreateAIAgent( + name: agentName, + model: deploymentName, + instructions: "You answer questions related to GitHub repositories only.", + tools: [.. mcpTools.Cast()]); + +string prompt = "Summarize the last four commits to the microsoft/semantic-kernel repository?"; + +Console.WriteLine($"Invoking agent '{agent.Name}' with prompt: {prompt} ..."); + +// Invoke the agent and output the text result. +Console.WriteLine(await agent.RunAsync(prompt)); + +// Clean up the agent after use. +await aiProjectClient.Agents.DeleteAgentAsync(agent.Name); diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step09_UsingMcpClientAsTools/README.md b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step09_UsingMcpClientAsTools/README.md new file mode 100644 index 0000000000..b2d923fc2f --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step09_UsingMcpClientAsTools/README.md @@ -0,0 +1,50 @@ +# Using MCP Client Tools with AI Agents + +This sample demonstrates how to use Model Context Protocol (MCP) client tools with AI agents, allowing agents to access tools provided by MCP servers. This sample uses the GitHub MCP server to provide tools for querying GitHub repositories. + +## What this sample demonstrates + +- Creating MCP clients to connect to MCP servers (GitHub server) +- Retrieving tools from MCP servers +- Using MCP tools with AI agents +- Running agents with MCP-provided function tools +- Managing agent lifecycle (creation and deletion) + +## Prerequisites + +Before you begin, ensure you have the following prerequisites: + +- .NET 10 SDK or later +- Azure Foundry service endpoint and deployment configured +- Azure CLI installed and authenticated (for Azure credential authentication) +- Node.js and npm installed (for running the GitHub MCP server) + +**Note**: This demo uses Azure CLI credentials for authentication. Make sure you're logged in with `az login` and have access to the Azure Foundry resource. For more information, see the [Azure CLI documentation](https://learn.microsoft.com/cli/azure/authenticate-azure-cli-interactively). + +Set the following environment variables: + +```powershell +$env:AZURE_FOUNDRY_PROJECT_ENDPOINT="https://your-foundry-service.services.ai.azure.com/api/projects/your-foundry-project" # Replace with your Azure Foundry resource endpoint +$env:AZURE_FOUNDRY_PROJECT_DEPLOYMENT_NAME="gpt-4o-mini" # Optional, defaults to gpt-4o-mini +``` + +## Run the sample + +Navigate to the FoundryAgents sample directory and run: + +```powershell +cd dotnet/samples/GettingStarted/FoundryAgents +dotnet run --project .\FoundryAgents_Step09_UsingMcpClientAsTools +``` + +## Expected behavior + +The sample will: + +1. Start the GitHub MCP server using `@modelcontextprotocol/server-github` +2. Create an MCP client to connect to the GitHub server +3. Retrieve the available tools from the GitHub MCP server +4. Create an agent named "AgentWithMCP" with the GitHub tools +5. Run the agent with a prompt to summarize the last four commits to the microsoft/semantic-kernel repository +6. The agent will use the GitHub MCP tools to query the repository information +7. Clean up resources by deleting the agent \ No newline at end of file diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step10_UsingImages/Assets/walkway.jpg b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step10_UsingImages/Assets/walkway.jpg new file mode 100644 index 0000000000..13ef1e1840 Binary files /dev/null and b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step10_UsingImages/Assets/walkway.jpg differ diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step10_UsingImages/FoundryAgents_Step10_UsingImages.csproj b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step10_UsingImages/FoundryAgents_Step10_UsingImages.csproj new file mode 100644 index 0000000000..53661ff199 --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step10_UsingImages/FoundryAgents_Step10_UsingImages.csproj @@ -0,0 +1,26 @@ + + + + Exe + net10.0 + + enable + enable + + + + + + + + + + + + + + Always + + + + diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step10_UsingImages/Program.cs b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step10_UsingImages/Program.cs new file mode 100644 index 0000000000..a799fe46fb --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step10_UsingImages/Program.cs @@ -0,0 +1,35 @@ +// Copyright (c) Microsoft. All rights reserved. + +// This sample shows how to use Image Multi-Modality with an AI agent. + +using Azure.AI.Projects; +using Azure.Identity; +using Microsoft.Agents.AI; +using Microsoft.Extensions.AI; + +string endpoint = Environment.GetEnvironmentVariable("AZURE_FOUNDRY_PROJECT_ENDPOINT") ?? throw new InvalidOperationException("AZURE_FOUNDRY_PROJECT_ENDPOINT is not set."); +string deploymentName = Environment.GetEnvironmentVariable("AZURE_FOUNDRY_PROJECT_DEPLOYMENT_NAME") ?? "gpt-4o"; + +const string VisionInstructions = "You are a helpful agent that can analyze images"; +const string VisionName = "VisionAgent"; + +// Get a client to create/retrieve/delete server side agents with Azure Foundry Agents. +AIProjectClient aiProjectClient = new(new Uri(endpoint), new AzureCliCredential()); + +// Define the agent you want to create. (Prompt Agent in this case) +AIAgent agent = aiProjectClient.CreateAIAgent(name: VisionName, model: deploymentName, instructions: VisionInstructions); + +ChatMessage message = new(ChatRole.User, [ + new TextContent("What do you see in this image?"), + new DataContent(File.ReadAllBytes("assets/walkway.jpg"), "image/jpeg") +]); + +AgentThread thread = agent.GetNewThread(); + +await foreach (AgentRunResponseUpdate update in agent.RunStreamingAsync(message, thread)) +{ + Console.WriteLine(update); +} + +// Cleanup by agent name removes the agent version created. +await aiProjectClient.Agents.DeleteAgentAsync(agent.Name); diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step10_UsingImages/README.md b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step10_UsingImages/README.md new file mode 100644 index 0000000000..d90f5cf208 --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step10_UsingImages/README.md @@ -0,0 +1,53 @@ +# Using Images with AI Agents + +This sample demonstrates how to use image multi-modality with an AI agent. It shows how to create a vision-enabled agent that can analyze and describe images using Azure Foundry Agents. + +## What this sample demonstrates + +- Creating a vision-enabled AI agent with image analysis capabilities +- Sending both text and image content to an agent in a single message +- Using `UriContent` for URI-referenced images +- Processing multimodal input (text + image) with an AI agent +- Managing agent lifecycle (creation and deletion) + +## Key features + +- **Vision Agent**: Creates an agent specifically instructed to analyze images +- **Multimodal Input**: Combines text questions with image URI in a single message +- **Azure Foundry Agents Integration**: Uses Azure Foundry Agents with vision capabilities + +## Prerequisites + +Before running this sample, ensure you have: + +1. An Azure OpenAI project set up +2. A compatible model deployment (e.g., gpt-4o) +3. Azure CLI installed and authenticated + +## Environment Variables + +Set the following environment variables: + +```powershell +$env:AZURE_FOUNDRY_PROJECT_ENDPOINT="https://your-resource.openai.azure.com/" # Replace with your Azure Foundry Project endpoint +$env:AZURE_FOUNDRY_PROJECT_DEPLOYMENT_NAME="gpt-4o" # Replace with your model deployment name (optional, defaults to gpt-4o) +``` + +## Run the sample + +Navigate to the FoundryAgents sample directory and run: + +```powershell +cd dotnet/samples/GettingStarted/FoundryAgents +dotnet run --project .\FoundryAgents_Step10_UsingImages +``` + +## Expected behavior + +The sample will: + +1. Create a vision-enabled agent named "VisionAgent" +2. Send a message containing both text ("What do you see in this image?") and a URI-referenced image of a green walkway (nature boardwalk) +3. The agent will analyze the image and provide a description +4. Clean up resources by deleting the agent + diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step11_AsFunctionTool/FoundryAgents_Step11_AsFunctionTool.csproj b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step11_AsFunctionTool/FoundryAgents_Step11_AsFunctionTool.csproj new file mode 100644 index 0000000000..54f37f1aa6 --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step11_AsFunctionTool/FoundryAgents_Step11_AsFunctionTool.csproj @@ -0,0 +1,21 @@ + + + + Exe + net10.0 + + enable + enable + 3afc9b74-af74-4d8e-ae96-fa1c511d11ac + + + + + + + + + + + + diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step11_AsFunctionTool/Program.cs b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step11_AsFunctionTool/Program.cs new file mode 100644 index 0000000000..9fb589f5ce --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step11_AsFunctionTool/Program.cs @@ -0,0 +1,47 @@ +// Copyright (c) Microsoft. All rights reserved. + +// This sample shows how to create and use an Azure Foundry Agents AI agent as a function tool. + +using System.ComponentModel; +using Azure.AI.Projects; +using Azure.Identity; +using Microsoft.Agents.AI; +using Microsoft.Extensions.AI; + +string endpoint = Environment.GetEnvironmentVariable("AZURE_FOUNDRY_PROJECT_ENDPOINT") ?? throw new InvalidOperationException("AZURE_FOUNDRY_PROJECT_ENDPOINT is not set."); +string deploymentName = Environment.GetEnvironmentVariable("AZURE_FOUNDRY_PROJECT_DEPLOYMENT_NAME") ?? "gpt-4o-mini"; + +const string WeatherInstructions = "You answer questions about the weather."; +const string WeatherName = "WeatherAgent"; +const string MainInstructions = "You are a helpful assistant who responds in French."; +const string MainName = "MainAgent"; + +[Description("Get the weather for a given location.")] +static string GetWeather([Description("The location to get the weather for.")] string location) + => $"The weather in {location} is cloudy with a high of 15°C."; + +// Get a client to create/retrieve/delete server side agents with Azure Foundry Agents. +AIProjectClient aiProjectClient = new(new Uri(endpoint), new AzureCliCredential()); + +// Create the weather agent with function tools. +AITool weatherTool = AIFunctionFactory.Create(GetWeather); +AIAgent weatherAgent = aiProjectClient.CreateAIAgent( + name: WeatherName, + model: deploymentName, + instructions: WeatherInstructions, + tools: [weatherTool]); + +// Create the main agent, and provide the weather agent as a function tool. +AIAgent agent = aiProjectClient.CreateAIAgent( + name: MainName, + model: deploymentName, + instructions: MainInstructions, + tools: [weatherAgent.AsAIFunction()]); + +// Invoke the agent and output the text result. +AgentThread thread = agent.GetNewThread(); +Console.WriteLine(await agent.RunAsync("What is the weather like in Amsterdam?", thread)); + +// Cleanup by agent name removes the agent versions created. +await aiProjectClient.Agents.DeleteAgentAsync(agent.Name); +await aiProjectClient.Agents.DeleteAgentAsync(weatherAgent.Name); diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step11_AsFunctionTool/README.md b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step11_AsFunctionTool/README.md new file mode 100644 index 0000000000..4b64b7e712 --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step11_AsFunctionTool/README.md @@ -0,0 +1,49 @@ +# Using AI Agents as Function Tools (Nested Agents) + +This sample demonstrates how to expose an AI agent as a function tool, enabling nested agent scenarios where one agent can invoke another agent as a tool. + +## What this sample demonstrates + +- Creating an AI agent that can be used as a function tool +- Wrapping an agent as an AIFunction +- Using nested agents where one agent calls another +- Managing multiple agent instances +- Managing agent lifecycle (creation and deletion) + +## Prerequisites + +Before you begin, ensure you have the following prerequisites: + +- .NET 10 SDK or later +- Azure Foundry service endpoint and deployment configured +- Azure CLI installed and authenticated (for Azure credential authentication) + +**Note**: This demo uses Azure CLI credentials for authentication. Make sure you're logged in with `az login` and have access to the Azure Foundry resource. For more information, see the [Azure CLI documentation](https://learn.microsoft.com/cli/azure/authenticate-azure-cli-interactively). + +Set the following environment variables: + +```powershell +$env:AZURE_FOUNDRY_PROJECT_ENDPOINT="https://your-foundry-service.services.ai.azure.com/api/projects/your-foundry-project" # Replace with your Azure Foundry resource endpoint +$env:AZURE_FOUNDRY_PROJECT_DEPLOYMENT_NAME="gpt-4o-mini" # Optional, defaults to gpt-4o-mini +``` + +## Run the sample + +Navigate to the FoundryAgents sample directory and run: + +```powershell +cd dotnet/samples/GettingStarted/FoundryAgents +dotnet run --project .\FoundryAgents_Step11_AsFunctionTool +``` + +## Expected behavior + +The sample will: + +1. Create a "JokerAgent" that tells jokes +2. Wrap the JokerAgent as a function tool +3. Create a "CoordinatorAgent" that has the JokerAgent as a function tool +4. Run the CoordinatorAgent with a prompt that triggers it to call the JokerAgent +5. The CoordinatorAgent will invoke the JokerAgent as a function tool +6. Clean up resources by deleting both agents + diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step12_Middleware/FoundryAgents_Step12_Middleware.csproj b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step12_Middleware/FoundryAgents_Step12_Middleware.csproj new file mode 100644 index 0000000000..9f29a8d7e6 --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step12_Middleware/FoundryAgents_Step12_Middleware.csproj @@ -0,0 +1,21 @@ + + + + Exe + net10.0 + + enable + enable + + + + + + + + + + + + + diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step12_Middleware/Program.cs b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step12_Middleware/Program.cs new file mode 100644 index 0000000000..0a00e9107c --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step12_Middleware/Program.cs @@ -0,0 +1,223 @@ +// Copyright (c) Microsoft. All rights reserved. + +// This sample shows multiple middleware layers working together with Azure Foundry Agents: +// agent run (PII filtering and guardrails), +// function invocation (logging and result overrides), and human-in-the-loop +// approval workflows for sensitive function calls. + +using System.ComponentModel; +using System.Text.RegularExpressions; +using Azure.AI.Projects; +using Azure.Identity; +using Microsoft.Agents.AI; +using Microsoft.Extensions.AI; + +// Get Azure AI Foundry configuration from environment variables +string endpoint = Environment.GetEnvironmentVariable("AZURE_FOUNDRY_PROJECT_ENDPOINT") ?? throw new InvalidOperationException("AZURE_FOUNDRY_PROJECT_ENDPOINT is not set."); +string deploymentName = System.Environment.GetEnvironmentVariable("AZURE_FOUNDRY_PROJECT_DEPLOYMENT_NAME") ?? "gpt-4o"; + +const string AssistantInstructions = "You are an AI assistant that helps people find information."; +const string AssistantName = "InformationAssistant"; + +// Get a client to create/retrieve/delete server side agents with Azure Foundry Agents. +AIProjectClient aiProjectClient = new(new Uri(endpoint), new AzureCliCredential()); + +[Description("Get the weather for a given location.")] +static string GetWeather([Description("The location to get the weather for.")] string location) + => $"The weather in {location} is cloudy with a high of 15°C."; + +[Description("The current datetime offset.")] +static string GetDateTime() + => DateTimeOffset.Now.ToString(); + +AITool dateTimeTool = AIFunctionFactory.Create(GetDateTime, name: nameof(GetDateTime)); +AITool getWeatherTool = AIFunctionFactory.Create(GetWeather, name: nameof(GetWeather)); + +// Define the agent you want to create. (Prompt Agent in this case) +AIAgent originalAgent = aiProjectClient.CreateAIAgent( + name: AssistantName, + model: deploymentName, + instructions: AssistantInstructions, + tools: [getWeatherTool, dateTimeTool]); + +// Adding middleware to the agent level +AIAgent middlewareEnabledAgent = originalAgent + .AsBuilder() + .Use(FunctionCallMiddleware) + .Use(FunctionCallOverrideWeather) + .Use(PIIMiddleware, null) + .Use(GuardrailMiddleware, null) + .Build(); + +AgentThread thread = middlewareEnabledAgent.GetNewThread(); + +Console.WriteLine("\n\n=== Example 1: Wording Guardrail ==="); +AgentRunResponse guardRailedResponse = await middlewareEnabledAgent.RunAsync("Tell me something harmful."); +Console.WriteLine($"Guard railed response: {guardRailedResponse}"); + +Console.WriteLine("\n\n=== Example 2: PII detection ==="); +AgentRunResponse piiResponse = await middlewareEnabledAgent.RunAsync("My name is John Doe, call me at 123-456-7890 or email me at john@something.com"); +Console.WriteLine($"Pii filtered response: {piiResponse}"); + +Console.WriteLine("\n\n=== Example 3: Agent function middleware ==="); + +// Agent function middleware support is limited to agents that wraps a upstream ChatClientAgent or derived from it. + +AgentRunResponse functionCallResponse = await middlewareEnabledAgent.RunAsync("What's the current time and the weather in Seattle?", thread); +Console.WriteLine($"Function calling response: {functionCallResponse}"); + +// Special per-request middleware agent. +Console.WriteLine("\n\n=== Example 4: Middleware with human in the loop function approval ==="); + +AIAgent humanInTheLoopAgent = aiProjectClient.CreateAIAgent( + name: "HumanInTheLoopAgent", + model: deploymentName, + instructions: "You are an Human in the loop testing AI assistant that helps people find information.", + + // Adding a function with approval required + tools: [new ApprovalRequiredAIFunction(AIFunctionFactory.Create(GetWeather, name: nameof(GetWeather)))]); + +// Using the ConsolePromptingApprovalMiddleware for a specific request to handle user approval during function calls. +AgentRunResponse response = await humanInTheLoopAgent + .AsBuilder() + .Use(ConsolePromptingApprovalMiddleware, null) + .Build() + .RunAsync("What's the current time and the weather in Seattle?"); + +Console.WriteLine($"HumanInTheLoopAgent agent middleware response: {response}"); + +// Function invocation middleware that logs before and after function calls. +async ValueTask FunctionCallMiddleware(AIAgent agent, FunctionInvocationContext context, Func> next, CancellationToken cancellationToken) +{ + Console.WriteLine($"Function Name: {context!.Function.Name} - Middleware 1 Pre-Invoke"); + var result = await next(context, cancellationToken); + Console.WriteLine($"Function Name: {context!.Function.Name} - Middleware 1 Post-Invoke"); + + return result; +} + +// Function invocation middleware that overrides the result of the GetWeather function. +async ValueTask FunctionCallOverrideWeather(AIAgent agent, FunctionInvocationContext context, Func> next, CancellationToken cancellationToken) +{ + Console.WriteLine($"Function Name: {context!.Function.Name} - Middleware 2 Pre-Invoke"); + + var result = await next(context, cancellationToken); + + if (context.Function.Name == nameof(GetWeather)) + { + // Override the result of the GetWeather function + result = "The weather is sunny with a high of 25°C."; + } + Console.WriteLine($"Function Name: {context!.Function.Name} - Middleware 2 Post-Invoke"); + return result; +} + +// This middleware redacts PII information from input and output messages. +async Task PIIMiddleware(IEnumerable messages, AgentThread? thread, AgentRunOptions? options, AIAgent innerAgent, CancellationToken cancellationToken) +{ + // Redact PII information from input messages + var filteredMessages = FilterMessages(messages); + Console.WriteLine("Pii Middleware - Filtered Messages Pre-Run"); + + var response = await innerAgent.RunAsync(filteredMessages, thread, options, cancellationToken).ConfigureAwait(false); + + // Redact PII information from output messages + response.Messages = FilterMessages(response.Messages); + + Console.WriteLine("Pii Middleware - Filtered Messages Post-Run"); + + return response; + + static IList FilterMessages(IEnumerable messages) + { + return messages.Select(m => new ChatMessage(m.Role, FilterPii(m.Text))).ToList(); + } + + static string FilterPii(string content) + { + // Regex patterns for PII detection (simplified for demonstration) + Regex[] piiPatterns = [ + new(@"\b\d{3}-\d{3}-\d{4}\b", RegexOptions.Compiled), // Phone number (e.g., 123-456-7890) + new(@"\b[\w\.-]+@[\w\.-]+\.\w+\b", RegexOptions.Compiled), // Email address + new(@"\b[A-Z][a-z]+\s[A-Z][a-z]+\b", RegexOptions.Compiled) // Full name (e.g., John Doe) + ]; + + foreach (var pattern in piiPatterns) + { + content = pattern.Replace(content, "[REDACTED: PII]"); + } + + return content; + } +} + +// This middleware enforces guardrails by redacting certain keywords from input and output messages. +async Task GuardrailMiddleware(IEnumerable messages, AgentThread? thread, AgentRunOptions? options, AIAgent innerAgent, CancellationToken cancellationToken) +{ + // Redact keywords from input messages + var filteredMessages = FilterMessages(messages); + + Console.WriteLine("Guardrail Middleware - Filtered messages Pre-Run"); + + // Proceed with the agent run + var response = await innerAgent.RunAsync(filteredMessages, thread, options, cancellationToken); + + // Redact keywords from output messages + response.Messages = FilterMessages(response.Messages); + + Console.WriteLine("Guardrail Middleware - Filtered messages Post-Run"); + + return response; + + List FilterMessages(IEnumerable messages) + { + return messages.Select(m => new ChatMessage(m.Role, FilterContent(m.Text))).ToList(); + } + + static string FilterContent(string content) + { + foreach (var keyword in new[] { "harmful", "illegal", "violence" }) + { + if (content.Contains(keyword, StringComparison.OrdinalIgnoreCase)) + { + return "[REDACTED: Forbidden content]"; + } + } + + return content; + } +} + +// This middleware handles Human in the loop console interaction for any user approval required during function calling. +async Task ConsolePromptingApprovalMiddleware(IEnumerable messages, AgentThread? thread, AgentRunOptions? options, AIAgent innerAgent, CancellationToken cancellationToken) +{ + AgentRunResponse response = await innerAgent.RunAsync(messages, thread, options, cancellationToken); + + List userInputRequests = response.UserInputRequests.ToList(); + + while (userInputRequests.Count > 0) + { + // Ask the user to approve each function call request. + // For simplicity, we are assuming here that only function approval requests are being made. + + // Pass the user input responses back to the agent for further processing. + response.Messages = userInputRequests + .OfType() + .Select(functionApprovalRequest => + { + Console.WriteLine($"The agent would like to invoke the following function, please reply Y to approve: Name {functionApprovalRequest.FunctionCall.Name}"); + bool approved = Console.ReadLine()?.Equals("Y", StringComparison.OrdinalIgnoreCase) ?? false; + return new ChatMessage(ChatRole.User, [functionApprovalRequest.CreateResponse(approved)]); + }) + .ToList(); + + response = await innerAgent.RunAsync(response.Messages, thread, options, cancellationToken); + + userInputRequests = response.UserInputRequests.ToList(); + } + + return response; +} + +// Cleanup by agent name removes the agent version created. +await aiProjectClient.Agents.DeleteAgentAsync(middlewareEnabledAgent.Name); diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step12_Middleware/README.md b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step12_Middleware/README.md new file mode 100644 index 0000000000..04192a2cc6 --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step12_Middleware/README.md @@ -0,0 +1,58 @@ +# Agent Middleware + +This sample demonstrates how to add middleware to intercept agent runs and function calls to implement cross-cutting concerns like logging, validation, and guardrails. + +## What This Sample Shows + +1. Azure Foundry Agents integration via `AIProjectClient` and `AzureCliCredential` +2. Agent run middleware (logging and monitoring) +3. Function invocation middleware (logging and overriding tool results) +4. Per-request agent run middleware +5. Per-request function pipeline with approval +6. Combining agent-level and per-request middleware + +## Function Invocation Middleware + +Not all agents support function invocation middleware. + +Attempting to use function middleware on agents that do not wrap a ChatClientAgent or derives from it will throw an InvalidOperationException. + +## Prerequisites + +Before you begin, ensure you have the following prerequisites: + +- .NET 10 SDK or later +- Azure Foundry service endpoint and deployment configured +- Azure CLI installed and authenticated (for Azure credential authentication) + +**Note**: This demo uses Azure CLI credentials for authentication. Make sure you're logged in with `az login` and have access to the Azure Foundry resource. For more information, see the [Azure CLI documentation](https://learn.microsoft.com/cli/azure/authenticate-azure-cli-interactively). + +Set the following environment variables: + +```powershell +$env:AZURE_FOUNDRY_PROJECT_ENDPOINT="https://your-foundry-service.services.ai.azure.com/api/projects/your-foundry-project" # Replace with your Azure Foundry resource endpoint +$env:AZURE_FOUNDRY_PROJECT_DEPLOYMENT_NAME="gpt-4o-mini" # Optional, defaults to gpt-4o-mini +``` + +## Running the Sample + +Navigate to the FoundryAgents sample directory and run: + +```powershell +cd dotnet/samples/GettingStarted/FoundryAgents +dotnet run --project .\FoundryAgents_Step12_Middleware +``` + +## Expected Behavior + +When you run this sample, you will see the following demonstrations: + +1. **Example 1: Wording Guardrail** - The agent receives a request for harmful content. The guardrail middleware intercepts the request and prevents the agent from responding to harmful prompts, returning a safe response instead. + +2. **Example 2: PII Detection** - The agent receives a message containing personally identifiable information (name, phone number, email). The PII middleware detects and filters this sensitive information before processing. + +3. **Example 3: Agent Function Middleware** - The agent uses function tools (GetDateTime and GetWeather) to answer a question about the current time and weather in Seattle. The function middleware logs the function calls and can override results if needed. + +4. **Example 4: Human-in-the-Loop Function Approval** - The agent attempts to call a weather function, but the approval middleware intercepts the call and prompts the user to approve or deny the function invocation before it executes. The user can respond with "Y" to approve or any other input to deny. + +Each example demonstrates how middleware can be used to implement cross-cutting concerns and control agent behavior at different levels (agent-level and per-request). diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step13_Plugins/FoundryAgents_Step13_Plugins.csproj b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step13_Plugins/FoundryAgents_Step13_Plugins.csproj new file mode 100644 index 0000000000..4a34560946 --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step13_Plugins/FoundryAgents_Step13_Plugins.csproj @@ -0,0 +1,22 @@ + + + + Exe + net10.0 + + enable + enable + $(NoWarn);CA1812 + + + + + + + + + + + + + diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step13_Plugins/Program.cs b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step13_Plugins/Program.cs new file mode 100644 index 0000000000..b55f38b66b --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step13_Plugins/Program.cs @@ -0,0 +1,139 @@ +// Copyright (c) Microsoft. All rights reserved. + +// This sample shows how to use plugins with an AI agent. Plugin classes can +// depend on other services that need to be injected. In this sample, the +// AgentPlugin class uses the WeatherProvider and CurrentTimeProvider classes +// to get weather and current time information. Both services are registered +// in the service collection and injected into the plugin. +// Plugin classes may have many methods, but only some are intended to be used +// as AI functions. The AsAITools method of the plugin class shows how to specify +// which methods should be exposed to the AI agent. + +using Azure.AI.Projects; +using Azure.Identity; +using Microsoft.Agents.AI; +using Microsoft.Extensions.AI; +using Microsoft.Extensions.DependencyInjection; + +string endpoint = Environment.GetEnvironmentVariable("AZURE_FOUNDRY_PROJECT_ENDPOINT") ?? throw new InvalidOperationException("AZURE_FOUNDRY_PROJECT_ENDPOINT is not set."); +string deploymentName = Environment.GetEnvironmentVariable("AZURE_FOUNDRY_PROJECT_DEPLOYMENT_NAME") ?? "gpt-4o-mini"; + +const string AssistantInstructions = "You are a helpful assistant that helps people find information."; +const string AssistantName = "PluginAssistant"; + +// Create a service collection to hold the agent plugin and its dependencies. +ServiceCollection services = new(); +services.AddSingleton(); +services.AddSingleton(); +services.AddSingleton(); // The plugin depends on WeatherProvider and CurrentTimeProvider registered above. + +IServiceProvider serviceProvider = services.BuildServiceProvider(); + +// Get a client to create/retrieve/delete server side agents with Azure Foundry Agents. +AIProjectClient aiProjectClient = new(new Uri(endpoint), new AzureCliCredential()); + +// Define the agent with plugin tools +// Define the agent you want to create. (Prompt Agent in this case) +AIAgent agent = aiProjectClient.CreateAIAgent( + name: AssistantName, + model: deploymentName, + instructions: AssistantInstructions, + tools: serviceProvider.GetRequiredService().AsAITools().ToList(), + services: serviceProvider); + +// Invoke the agent and output the text result. +AgentThread thread = agent.GetNewThread(); +Console.WriteLine(await agent.RunAsync("Tell me current time and weather in Seattle.", thread)); + +// Cleanup by agent name removes the agent version created. +await aiProjectClient.Agents.DeleteAgentAsync(agent.Name); + +/// +/// The agent plugin that provides weather and current time information. +/// +/// The weather provider to get weather information. +internal sealed class AgentPlugin(WeatherProvider weatherProvider) +{ + /// + /// Gets the weather information for the specified location. + /// + /// + /// This method demonstrates how to use the dependency that was injected into the plugin class. + /// + /// The location to get the weather for. + /// The weather information for the specified location. + public string GetWeather(string location) + { + return weatherProvider.GetWeather(location); + } + + /// + /// Gets the current date and time for the specified location. + /// + /// + /// This method demonstrates how to resolve a dependency using the service provider passed to the method. + /// + /// The service provider to resolve the . + /// The location to get the current time for. + /// The current date and time as a . + public DateTimeOffset GetCurrentTime(IServiceProvider sp, string location) + { + // Resolve the CurrentTimeProvider from the service provider + CurrentTimeProvider currentTimeProvider = sp.GetRequiredService(); + + return currentTimeProvider.GetCurrentTime(location); + } + + /// + /// Returns the functions provided by this plugin. + /// + /// + /// In real world scenarios, a class may have many methods and only a subset of them may be intended to be exposed as AI functions. + /// This method demonstrates how to explicitly specify which methods should be exposed to the AI agent. + /// + /// The functions provided by this plugin. + public IEnumerable AsAITools() + { + yield return AIFunctionFactory.Create(this.GetWeather); + yield return AIFunctionFactory.Create(this.GetCurrentTime); + } +} + +/// +/// The weather provider that returns weather information. +/// +internal sealed class WeatherProvider +{ + /// + /// Gets the weather information for the specified location. + /// + /// + /// The weather information is hardcoded for demonstration purposes. + /// In a real application, this could call a weather API to get actual weather data. + /// + /// The location to get the weather for. + /// The weather information for the specified location. + public string GetWeather(string location) + { + return $"The weather in {location} is cloudy with a high of 15°C."; + } +} + +/// +/// Provides the current date and time. +/// +/// +/// This class returns the current date and time using the system's clock. +/// +internal sealed class CurrentTimeProvider +{ + /// + /// Gets the current date and time. + /// + /// The location to get the current time for (not used in this implementation). + /// The current date and time as a . + public DateTimeOffset GetCurrentTime(string location) + { + return DateTimeOffset.Now; + } +} diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step13_Plugins/README.md b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step13_Plugins/README.md new file mode 100644 index 0000000000..0aeccf5789 --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step13_Plugins/README.md @@ -0,0 +1,49 @@ +# Using Plugins with AI Agents + +This sample demonstrates how to use plugins with AI agents, where plugins are services registered in dependency injection that expose methods as AI function tools. + +## What this sample demonstrates + +- Creating plugin services with methods to expose as tools +- Using AsAITools() to selectively expose plugin methods +- Registering plugins in dependency injection +- Using plugins with AI agents +- Managing agent lifecycle (creation and deletion) + +## Prerequisites + +Before you begin, ensure you have the following prerequisites: + +- .NET 10 SDK or later +- Azure Foundry service endpoint and deployment configured +- Azure CLI installed and authenticated (for Azure credential authentication) + +**Note**: This demo uses Azure CLI credentials for authentication. Make sure you're logged in with `az login` and have access to the Azure Foundry resource. For more information, see the [Azure CLI documentation](https://learn.microsoft.com/cli/azure/authenticate-azure-cli-interactively). + +Set the following environment variables: + +```powershell +$env:AZURE_FOUNDRY_PROJECT_ENDPOINT="https://your-foundry-service.services.ai.azure.com/api/projects/your-foundry-project" # Replace with your Azure Foundry resource endpoint +$env:AZURE_FOUNDRY_PROJECT_DEPLOYMENT_NAME="gpt-4o-mini" # Optional, defaults to gpt-4o-mini +``` + +## Run the sample + +Navigate to the FoundryAgents sample directory and run: + +```powershell +cd dotnet/samples/GettingStarted/FoundryAgents +dotnet run --project .\FoundryAgents_Step13_Plugins +``` + +## Expected behavior + +The sample will: + +1. Create a plugin service with methods to expose as tools +2. Register the plugin in dependency injection +3. Create an agent named "PluginAgent" with the plugin methods as function tools +4. Run the agent with a prompt that triggers it to call plugin methods +5. The agent will invoke the plugin methods to retrieve information +6. Clean up resources by deleting the agent + diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step14_CodeInterpreter/FoundryAgents_Step14_CodeInterpreter.csproj b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step14_CodeInterpreter/FoundryAgents_Step14_CodeInterpreter.csproj new file mode 100644 index 0000000000..4a34560946 --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step14_CodeInterpreter/FoundryAgents_Step14_CodeInterpreter.csproj @@ -0,0 +1,22 @@ + + + + Exe + net10.0 + + enable + enable + $(NoWarn);CA1812 + + + + + + + + + + + + + diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step14_CodeInterpreter/Program.cs b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step14_CodeInterpreter/Program.cs new file mode 100644 index 0000000000..0f6f6ef2d9 --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step14_CodeInterpreter/Program.cs @@ -0,0 +1,90 @@ +// Copyright (c) Microsoft. All rights reserved. + +// This sample shows how to use Code Interpreter Tool with AI Agents. + +using System.Text; +using Azure.AI.Projects; +using Azure.AI.Projects.OpenAI; +using Azure.Identity; +using Microsoft.Agents.AI; +using Microsoft.Extensions.AI; +using OpenAI.Assistants; +using OpenAI.Responses; + +string endpoint = Environment.GetEnvironmentVariable("AZURE_FOUNDRY_PROJECT_ENDPOINT") ?? throw new InvalidOperationException("AZURE_FOUNDRY_PROJECT_ENDPOINT is not set."); +string deploymentName = Environment.GetEnvironmentVariable("AZURE_FOUNDRY_PROJECT_DEPLOYMENT_NAME") ?? "gpt-4o-mini"; + +const string AgentInstructions = "You are a personal math tutor. When asked a math question, write and run code using the python tool to answer the question."; +const string AgentNameMEAI = "CoderAgent-MEAI"; +const string AgentNameNative = "CoderAgent-NATIVE"; + +// Get a client to create/retrieve/delete server side agents with Azure Foundry Agents. +AIProjectClient aiProjectClient = new(new Uri(endpoint), new AzureCliCredential()); + +// Option 1 - Using HostedCodeInterpreterTool + AgentOptions (MEAI + AgentFramework) +// Create the server side agent version +AIAgent agentOption1 = await aiProjectClient.CreateAIAgentAsync( + model: deploymentName, + name: AgentNameMEAI, + instructions: AgentInstructions, + tools: [new HostedCodeInterpreterTool() { Inputs = [] }]); + +// Option 2 - Using PromptAgentDefinition SDK native type +// Create the server side agent version +AIAgent agentOption2 = await aiProjectClient.CreateAIAgentAsync( + name: AgentNameNative, + creationOptions: new AgentVersionCreationOptions( + new PromptAgentDefinition(model: deploymentName) + { + Instructions = AgentInstructions, + Tools = { + ResponseTool.CreateCodeInterpreterTool( + new CodeInterpreterToolContainer( + CodeInterpreterToolContainerConfiguration.CreateAutomaticContainerConfiguration(fileIds: []) + ) + ), + } + }) +); + +// Either invoke option1 or option2 agent, should have same result +// Option 1 +AgentRunResponse response = await agentOption1.RunAsync("I need to solve the equation sin(x) + x^2 = 42"); + +// Option 2 +// AgentRunResponse response = await agentOption2.RunAsync("I need to solve the equation sin(x) + x^2 = 42"); + +// Get the CodeInterpreterToolCallContent +CodeInterpreterToolCallContent? toolCallContent = response.Messages.SelectMany(m => m.Contents).OfType().FirstOrDefault(); +if (toolCallContent?.Inputs is not null) +{ + DataContent? codeInput = toolCallContent.Inputs.OfType().FirstOrDefault(); + if (codeInput?.HasTopLevelMediaType("text") ?? false) + { + Console.WriteLine($"Code Input: {Encoding.UTF8.GetString(codeInput.Data.ToArray()) ?? "Not available"}"); + } +} + +// Get the CodeInterpreterToolResultContent +CodeInterpreterToolResultContent? toolResultContent = response.Messages.SelectMany(m => m.Contents).OfType().FirstOrDefault(); +if (toolResultContent?.Outputs is not null && toolResultContent.Outputs.OfType().FirstOrDefault() is { } resultOutput) +{ + Console.WriteLine($"Code Tool Result: {resultOutput.Text}"); +} + +// Getting any annotations generated by the tool +foreach (AIAnnotation annotation in response.Messages.SelectMany(m => m.Contents).SelectMany(C => C.Annotations ?? [])) +{ + if (annotation.RawRepresentation is TextAnnotationUpdate citationAnnotation) + { + Console.WriteLine($$""" + File Id: {{citationAnnotation.OutputFileId}} + Text to Replace: {{citationAnnotation.TextToReplace}} + Filename: {{Path.GetFileName(citationAnnotation.TextToReplace)}} + """); + } +} + +// Cleanup by agent name removes the agent version created. +await aiProjectClient.Agents.DeleteAgentAsync(agentOption1.Name); +await aiProjectClient.Agents.DeleteAgentAsync(agentOption2.Name); diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step14_CodeInterpreter/README.md b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step14_CodeInterpreter/README.md new file mode 100644 index 0000000000..a3dd4d50b9 --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step14_CodeInterpreter/README.md @@ -0,0 +1,53 @@ +# Using Code Interpreter with AI Agents + +This sample demonstrates how to use the code interpreter tool with AI agents. The code interpreter allows agents to write and execute Python code to solve problems, perform calculations, and analyze data. + +## What this sample demonstrates + +- Creating agents with code interpreter capabilities +- Using HostedCodeInterpreterTool (MEAI abstraction) +- Using native SDK code interpreter tools (ResponseTool.CreateCodeInterpreterTool) +- Extracting code inputs and results from agent responses +- Handling code interpreter annotations +- Managing agent lifecycle (creation and deletion) + +## Prerequisites + +Before you begin, ensure you have the following prerequisites: + +- .NET 10 SDK or later +- Azure Foundry service endpoint and deployment configured +- Azure CLI installed and authenticated (for Azure credential authentication) + +**Note**: This demo uses Azure CLI credentials for authentication. Make sure you're logged in with `az login` and have access to the Azure Foundry resource. For more information, see the [Azure CLI documentation](https://learn.microsoft.com/cli/azure/authenticate-azure-cli-interactively). + +Set the following environment variables: + +```powershell +$env:AZURE_FOUNDRY_PROJECT_ENDPOINT="https://your-foundry-service.services.ai.azure.com/api/projects/your-foundry-project" # Replace with your Azure Foundry resource endpoint +$env:AZURE_FOUNDRY_PROJECT_DEPLOYMENT_NAME="gpt-4o-mini" # Optional, defaults to gpt-4o-mini +``` + +## Run the sample + +Navigate to the FoundryAgents sample directory and run: + +```powershell +cd dotnet/samples/GettingStarted/FoundryAgents +dotnet run --project .\FoundryAgents_Step14_CodeInterpreter +``` + +## Expected behavior + +The sample will: + +1. Create two agents with code interpreter capabilities: + - Option 1: Using HostedCodeInterpreterTool (MEAI abstraction) + - Option 2: Using native SDK code interpreter tools +2. Run the agent with a mathematical problem: "I need to solve the equation sin(x) + x^2 = 42" +3. The agent will use the code interpreter to write and execute Python code to solve the equation +4. Extract and display the code that was executed +5. Display the results from the code execution +6. Display any annotations generated by the code interpreter tool +7. Clean up resources by deleting both agents + diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step15_ComputerUse/Assets/cua_browser_search.png b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step15_ComputerUse/Assets/cua_browser_search.png new file mode 100644 index 0000000000..5984b95cb3 Binary files /dev/null and b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step15_ComputerUse/Assets/cua_browser_search.png differ diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step15_ComputerUse/Assets/cua_search_results.png b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step15_ComputerUse/Assets/cua_search_results.png new file mode 100644 index 0000000000..ed3ab3d8d4 Binary files /dev/null and b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step15_ComputerUse/Assets/cua_search_results.png differ diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step15_ComputerUse/Assets/cua_search_typed.png b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step15_ComputerUse/Assets/cua_search_typed.png new file mode 100644 index 0000000000..04d76e2075 Binary files /dev/null and b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step15_ComputerUse/Assets/cua_search_typed.png differ diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step15_ComputerUse/ComputerUseUtil.cs b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step15_ComputerUse/ComputerUseUtil.cs new file mode 100644 index 0000000000..1ee421b465 --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step15_ComputerUse/ComputerUseUtil.cs @@ -0,0 +1,98 @@ +// Copyright (c) Microsoft. All rights reserved. + +using OpenAI.Responses; + +namespace Demo.ComputerUse; + +/// +/// Enum for tracking the state of the simulated web search flow. +/// +internal enum SearchState +{ + Initial, // Browser search page + Typed, // Text entered in search box + PressedEnter // Enter key pressed, transitioning to results +} + +internal static class ComputerUseUtil +{ + /// + /// Load and convert screenshot images to base64 data URLs. + /// + internal static Dictionary LoadScreenshotAssets() + { + string baseDir = Path.Combine(AppDomain.CurrentDomain.BaseDirectory, "Assets"); + + ReadOnlySpan<(string key, string fileName)> screenshotFiles = + [ + ("browser_search", "cua_browser_search.png"), + ("search_typed", "cua_search_typed.png"), + ("search_results", "cua_search_results.png") + ]; + + Dictionary screenshots = []; + foreach (var (key, fileName) in screenshotFiles) + { + string fullPath = Path.GetFullPath(Path.Combine(baseDir, fileName)); + screenshots[key] = File.ReadAllBytes(fullPath); + } + + return screenshots; + } + + /// + /// Process a computer action and simulate its execution. + /// + internal static (SearchState CurrentState, byte[] ImageBytes) HandleComputerActionAndTakeScreenshot( + ComputerCallAction action, + SearchState currentState, + Dictionary screenshots) + { + Console.WriteLine($"Simulating the execution of computer action: {action.Kind}"); + + SearchState newState = DetermineNextState(action, currentState); + string imageKey = GetImageKey(newState); + + return (newState, screenshots[imageKey]); + } + + private static SearchState DetermineNextState(ComputerCallAction action, SearchState currentState) + { + string actionType = action.Kind.ToString(); + + if (actionType.Equals("type", StringComparison.OrdinalIgnoreCase) && action.TypeText is not null) + { + return SearchState.Typed; + } + + if (IsEnterKeyAction(action, actionType)) + { + Console.WriteLine(" -> Detected ENTER key press"); + return SearchState.PressedEnter; + } + + if (actionType.Equals("click", StringComparison.OrdinalIgnoreCase) && currentState == SearchState.Typed) + { + Console.WriteLine(" -> Detected click after typing"); + return SearchState.PressedEnter; + } + + return currentState; + } + + private static bool IsEnterKeyAction(ComputerCallAction action, string actionType) + { + return (actionType.Equals("key", StringComparison.OrdinalIgnoreCase) || + actionType.Equals("keypress", StringComparison.OrdinalIgnoreCase)) && + action.KeyPressKeyCodes is not null && + (action.KeyPressKeyCodes.Contains("Return", StringComparer.OrdinalIgnoreCase) || + action.KeyPressKeyCodes.Contains("Enter", StringComparer.OrdinalIgnoreCase)); + } + + private static string GetImageKey(SearchState state) => state switch + { + SearchState.PressedEnter => "search_results", + SearchState.Typed => "search_typed", + _ => "browser_search" + }; +} diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step15_ComputerUse/FoundryAgents_Step15_ComputerUse.csproj b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step15_ComputerUse/FoundryAgents_Step15_ComputerUse.csproj new file mode 100644 index 0000000000..041c72c43e --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step15_ComputerUse/FoundryAgents_Step15_ComputerUse.csproj @@ -0,0 +1,33 @@ + + + + Exe + net10.0 + + enable + enable + $(NoWarn);OPENAICUA001 + + + + + + + + + + + + + + Always + + + Always + + + Always + + + + diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step15_ComputerUse/Program.cs b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step15_ComputerUse/Program.cs new file mode 100644 index 0000000000..05fb39bbf4 --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step15_ComputerUse/Program.cs @@ -0,0 +1,174 @@ +// Copyright (c) Microsoft. All rights reserved. + +// This sample shows how to use Computer Use Tool with AI Agents. + +using Azure.AI.Projects; +using Azure.AI.Projects.OpenAI; +using Azure.Identity; +using Microsoft.Agents.AI; +using Microsoft.Extensions.AI; +using OpenAI.Responses; + +namespace Demo.ComputerUse; + +internal sealed class Program +{ + private static async Task Main(string[] args) + { + string endpoint = Environment.GetEnvironmentVariable("AZURE_FOUNDRY_PROJECT_ENDPOINT") ?? throw new InvalidOperationException("AZURE_FOUNDRY_PROJECT_ENDPOINT is not set."); + string deploymentName = Environment.GetEnvironmentVariable("AZURE_FOUNDRY_PROJECT_DEPLOYMENT_NAME") ?? "computer-use-preview"; + + // Get a client to create/retrieve/delete server side agents with Azure Foundry Agents. + AIProjectClient aiProjectClient = new(new Uri(endpoint), new AzureCliCredential()); + const string AgentInstructions = @" + You are a computer automation assistant. + + Be direct and efficient. When you reach the search results page, read and describe the actual search result titles and descriptions you can see. + "; + + const string AgentNameMEAI = "ComputerAgent-MEAI"; + const string AgentNameNative = "ComputerAgent-NATIVE"; + + // Option 1 - Using ComputerUseTool + AgentOptions (MEAI + AgentFramework) + // Create AIAgent directly + AIAgent agentOption1 = await aiProjectClient.CreateAIAgentAsync( + name: AgentNameMEAI, + model: deploymentName, + instructions: AgentInstructions, + description: "Computer automation agent with screen interaction capabilities.", + tools: [ + ResponseTool.CreateComputerTool(ComputerToolEnvironment.Browser, 1026, 769).AsAITool(), + ]); + + // Option 2 - Using PromptAgentDefinition SDK native type + // Create the server side agent version + AIAgent agentOption2 = await aiProjectClient.CreateAIAgentAsync( + name: AgentNameNative, + creationOptions: new AgentVersionCreationOptions( + new PromptAgentDefinition(model: deploymentName) + { + Instructions = AgentInstructions, + Tools = { ResponseTool.CreateComputerTool( + environment: new ComputerToolEnvironment("windows"), + displayWidth: 1026, + displayHeight: 769) } + }) + ); + + // Either invoke option1 or option2 agent, should have same result + // Option 1 + await InvokeComputerUseAgentAsync(agentOption1); + + // Option 2 + //await InvokeComputerUseAgentAsync(agentOption2); + + // Cleanup by agent name removes the agent version created. + await aiProjectClient.Agents.DeleteAgentAsync(agentOption1.Name); + await aiProjectClient.Agents.DeleteAgentAsync(agentOption2.Name); + } + + private static async Task InvokeComputerUseAgentAsync(AIAgent agent) + { + // Load screenshot assets + Dictionary screenshots = ComputerUseUtil.LoadScreenshotAssets(); + + ChatOptions chatOptions = new(); + ResponseCreationOptions responseCreationOptions = new() + { + TruncationMode = ResponseTruncationMode.Auto + }; + chatOptions.RawRepresentationFactory = (_) => responseCreationOptions; + ChatClientAgentRunOptions runOptions = new(chatOptions) + { + AllowBackgroundResponses = true, + }; + + AgentThread thread = agent.GetNewThread(); + + ChatMessage message = new(ChatRole.User, [ + new TextContent("I need you to help me search for 'OpenAI news'. Please type 'OpenAI news' and submit the search. Once you see search results, the task is complete."), + new DataContent(new BinaryData(screenshots["browser_search"]), "image/png") + ]); + + // Initial request with screenshot - start with Bing search page + Console.WriteLine("Starting computer automation session (initial screenshot: cua_browser_search.png)..."); + + AgentRunResponse runResponse = await agent.RunAsync(message, thread: thread, options: runOptions); + + // Main interaction loop + const int MaxIterations = 10; + int iteration = 0; + // Initialize state machine + SearchState currentState = SearchState.Initial; + string initialCallId = string.Empty; + + while (true) + { + // Poll until the response is complete. + while (runResponse.ContinuationToken is { } token) + { + // Wait before polling again. + await Task.Delay(TimeSpan.FromSeconds(2)); + + // Continue with the token. + runOptions.ContinuationToken = token; + + runResponse = await agent.RunAsync(thread, runOptions); + } + + Console.WriteLine($"Agent response received (ID: {runResponse.ResponseId})"); + + if (iteration >= MaxIterations) + { + Console.WriteLine($"\nReached maximum iterations ({MaxIterations}). Stopping."); + break; + } + + iteration++; + Console.WriteLine($"\n--- Iteration {iteration} ---"); + + // Check for computer calls in the response + IEnumerable computerCallResponseItems = runResponse.Messages + .SelectMany(x => x.Contents) + .Where(c => c.RawRepresentation is ComputerCallResponseItem and not null) + .Select(c => (ComputerCallResponseItem)c.RawRepresentation!); + + ComputerCallResponseItem? firstComputerCall = computerCallResponseItems.FirstOrDefault(); + if (firstComputerCall is null) + { + Console.WriteLine("No computer call actions found. Ending interaction."); + Console.WriteLine($"Final Response: {runResponse}"); + break; + } + + // Process the first computer call response + ComputerCallAction action = firstComputerCall.Action; + string currentCallId = firstComputerCall.CallId; + + // Set the initial computer call ID for tracking and subsequent responses. + if (string.IsNullOrEmpty(initialCallId)) + { + initialCallId = currentCallId; + } + + Console.WriteLine($"Processing computer call (ID: {currentCallId})"); + + // Simulate executing the action and taking a screenshot + (SearchState CurrentState, byte[] ImageBytes) screenInfo = ComputerUseUtil.HandleComputerActionAndTakeScreenshot(action, currentState, screenshots); + currentState = screenInfo.CurrentState; + + Console.WriteLine("Sending action result back to agent..."); + + AIContent content = new() + { + RawRepresentation = new ComputerCallOutputResponseItem( + initialCallId, + output: ComputerCallOutput.CreateScreenshotOutput(new BinaryData(screenInfo.ImageBytes), "image/png")) + }; + + // Follow-up message with action result and new screenshot + message = new(ChatRole.User, [content]); + runResponse = await agent.RunAsync(message, thread: thread, options: runOptions); + } + } +} diff --git a/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step15_ComputerUse/README.md b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step15_ComputerUse/README.md new file mode 100644 index 0000000000..4686ec5984 --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/FoundryAgents_Step15_ComputerUse/README.md @@ -0,0 +1,55 @@ +# Using Computer Use Tool with AI Agents + +This sample demonstrates how to use the computer use tool with AI agents. The computer use tool allows agents to interact with a computer environment by viewing the screen, controlling the mouse and keyboard, and performing various actions to help complete tasks. + +## What this sample demonstrates + +- Creating agents with computer use capabilities +- Using HostedComputerTool (MEAI abstraction) +- Using native SDK computer use tools (ResponseTool.CreateComputerTool) +- Extracting computer action information from agent responses +- Handling computer tool results (text output and screenshots) +- Managing agent lifecycle (creation and deletion) + +## Prerequisites + +Before you begin, ensure you have the following prerequisites: + +- .NET 10 SDK or later +- Azure Foundry service endpoint and deployment configured +- Azure CLI installed and authenticated (for Azure credential authentication) + +**Note**: This demo uses Azure CLI credentials for authentication. Make sure you're logged in with `az login` and have access to the Azure Foundry resource. For more information, see the [Azure CLI documentation](https://learn.microsoft.com/cli/azure/authenticate-azure-cli-interactively). + +Set the following environment variables: + +```powershell +$env:AZURE_FOUNDRY_PROJECT_ENDPOINT="https://your-foundry-service.services.ai.azure.com/api/projects/your-foundry-project" # Replace with your Azure Foundry resource endpoint +$env:AZURE_FOUNDRY_PROJECT_DEPLOYMENT_NAME="computer-use-preview" # Optional, defaults to computer-use-preview +``` + +## Run the sample + +Navigate to the FoundryAgents sample directory and run: + +```powershell +cd dotnet/samples/GettingStarted/FoundryAgents +dotnet run --project .\FoundryAgents_Step15_ComputerUse +``` + +## Expected behavior + +The sample will: + +1. Create two agents with computer use capabilities: + - Option 1: Using HostedComputerTool (MEAI abstraction) + - Option 2: Using native SDK computer use tools +2. Run the agent with a task: "I need you to help me search for 'OpenAI news'. Please type 'OpenAI news' and submit the search. Once you see search results, the task is complete." +3. The agent will use the computer use tool to: + - Interpret the screenshots + - Issue action requests based on the task + - Analyze the search results for "OpenAI news" from the screenshots. +4. Extract and display the computer actions performed +5. Display the results from the computer tool execution +6. Display the final response from the agent +7. Clean up resources by deleting both agents diff --git a/dotnet/samples/GettingStarted/FoundryAgents/README.md b/dotnet/samples/GettingStarted/FoundryAgents/README.md new file mode 100644 index 0000000000..daeb2db8df --- /dev/null +++ b/dotnet/samples/GettingStarted/FoundryAgents/README.md @@ -0,0 +1,91 @@ +# Getting started with Foundry Agents + +The getting started with Foundry Agents samples demonstrate the fundamental concepts and functionalities +of Azure Foundry Agents and can be used with Azure Foundry as the AI provider. + +These samples showcase how to work with agents managed through Azure Foundry, including agent creation, +versioning, multi-turn conversations, and advanced features like code interpretation and computer use. + +## Classic vs New Foundry Agents + +> [!NOTE] +> Recently, Azure Foundry introduced a new and improved experience for creating and managing AI agents, which is the target of these samples. + +For more information about the previous classic agents and for what's new in Foundry Agents, see the [Foundry Agents migration documentation](https://learn.microsoft.com/en-us/azure/ai-foundry/agents/how-to/migrate?view=foundry). + +For a sample demonstrating how to use classic Foundry Agents, see the following: [Agent with Azure AI Persistent](../AgentProviders/Agent_With_AzureAIAgentsPersistent/README.md). + +## Getting started with Foundry Agents prerequisites + +Before you begin, ensure you have the following prerequisites: + +- .NET 10 SDK or later +- Azure Foundry service endpoint and project configured +- Azure CLI installed and authenticated (for Azure credential authentication) + +**Note**: These samples use Azure Foundry Agents. For more information, see [Azure AI Foundry documentation](https://learn.microsoft.com/en-us/azure/ai-foundry/). + +**Note**: These samples use Azure CLI credentials for authentication. Make sure you're logged in with `az login` and have access to the Azure Foundry resource. For more information, see the [Azure CLI documentation](https://learn.microsoft.com/cli/azure/authenticate-azure-cli-interactively). + +## Samples + +|Sample|Description| +|---|---| +|[Basics](./FoundryAgents_Step01.1_Basics/)|This sample demonstrates how to create and manage AI agents with versioning| +|[Running a simple agent](./FoundryAgents_Step01.2_Running/)|This sample demonstrates how to create and run a basic Foundry agent| +|[Multi-turn conversation](./FoundryAgents_Step02_MultiturnConversation/)|This sample demonstrates how to implement a multi-turn conversation with a Foundry agent| +|[Using function tools](./FoundryAgents_Step03_UsingFunctionTools/)|This sample demonstrates how to use function tools with a Foundry agent| +|[Using function tools with approvals](./FoundryAgents_Step04_UsingFunctionToolsWithApprovals/)|This sample demonstrates how to use function tools where approvals require human in the loop approvals before execution| +|[Structured output](./FoundryAgents_Step05_StructuredOutput/)|This sample demonstrates how to use structured output with a Foundry agent| +|[Persisted conversations](./FoundryAgents_Step06_PersistedConversations/)|This sample demonstrates how to persist conversations and reload them later| +|[Observability](./FoundryAgents_Step07_Observability/)|This sample demonstrates how to add telemetry to a Foundry agent| +|[Dependency injection](./FoundryAgents_Step08_DependencyInjection/)|This sample demonstrates how to add and resolve a Foundry agent with a dependency injection container| +|[Using MCP client as tools](./FoundryAgents_Step09_UsingMcpClientAsTools/)|This sample demonstrates how to use MCP clients as tools with a Foundry agent| +|[Using images](./FoundryAgents_Step10_UsingImages/)|This sample demonstrates how to use image multi-modality with a Foundry agent| +|[Exposing as a function tool](./FoundryAgents_Step11_AsFunctionTool/)|This sample demonstrates how to expose a Foundry agent as a function tool| +|[Using middleware](./FoundryAgents_Step12_Middleware/)|This sample demonstrates how to use middleware with a Foundry agent| +|[Using plugins](./FoundryAgents_Step13_Plugins/)|This sample demonstrates how to use plugins with a Foundry agent| +|[Code interpreter](./FoundryAgents_Step14_CodeInterpreter/)|This sample demonstrates how to use the code interpreter tool with a Foundry agent| +|[Computer use](./FoundryAgents_Step15_ComputerUse/)|This sample demonstrates how to use computer use capabilities with a Foundry agent| + +## Running the samples from the console + +To run the samples, navigate to the desired sample directory, e.g. + +```powershell +cd FoundryAgents_Step01.2_Running +``` + +Set the following environment variables: + +```powershell +$env:AZURE_FOUNDRY_PROJECT_ENDPOINT="https://your-foundry-service.services.ai.azure.com/api/projects/your-foundry-project" # Replace with your Azure Foundry resource endpoint +$env:AZURE_FOUNDRY_PROJECT_DEPLOYMENT_NAME="gpt-4o-mini" # Optional, defaults to gpt-4o-mini +``` + +If the variables are not set, you will be prompted for the values when running the samples. + +Execute the following command to build the sample: + +```powershell +dotnet build +``` + +Execute the following command to run the sample: + +```powershell +dotnet run --no-build +``` + +Or just build and run in one step: + +```powershell +dotnet run +``` + +## Running the samples from Visual Studio + +Open the solution in Visual Studio and set the desired sample project as the startup project. Then, run the project using the built-in debugger or by pressing `F5`. + +You will be prompted for any required environment variables if they are not already set. + diff --git a/dotnet/samples/GettingStarted/ModelContextProtocol/Agent_MCP_Server/Agent_MCP_Server.csproj b/dotnet/samples/GettingStarted/ModelContextProtocol/Agent_MCP_Server/Agent_MCP_Server.csproj index c5e06bc382..aa73860c14 100644 --- a/dotnet/samples/GettingStarted/ModelContextProtocol/Agent_MCP_Server/Agent_MCP_Server.csproj +++ b/dotnet/samples/GettingStarted/ModelContextProtocol/Agent_MCP_Server/Agent_MCP_Server.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable @@ -13,7 +13,6 @@ - diff --git a/dotnet/samples/GettingStarted/ModelContextProtocol/Agent_MCP_Server/Program.cs b/dotnet/samples/GettingStarted/ModelContextProtocol/Agent_MCP_Server/Program.cs index 1a0d236961..774c33ed58 100644 --- a/dotnet/samples/GettingStarted/ModelContextProtocol/Agent_MCP_Server/Program.cs +++ b/dotnet/samples/GettingStarted/ModelContextProtocol/Agent_MCP_Server/Program.cs @@ -7,7 +7,7 @@ using Microsoft.Agents.AI; using Microsoft.Extensions.AI; using ModelContextProtocol.Client; -using OpenAI; +using OpenAI.Chat; var endpoint = Environment.GetEnvironmentVariable("AZURE_OPENAI_ENDPOINT") ?? throw new InvalidOperationException("AZURE_OPENAI_ENDPOINT is not set."); var deploymentName = Environment.GetEnvironmentVariable("AZURE_OPENAI_DEPLOYMENT_NAME") ?? "gpt-4o-mini"; diff --git a/dotnet/samples/GettingStarted/ModelContextProtocol/Agent_MCP_Server_Auth/Agent_MCP_Server_Auth.csproj b/dotnet/samples/GettingStarted/ModelContextProtocol/Agent_MCP_Server_Auth/Agent_MCP_Server_Auth.csproj index 389b504c50..46c1306149 100644 --- a/dotnet/samples/GettingStarted/ModelContextProtocol/Agent_MCP_Server_Auth/Agent_MCP_Server_Auth.csproj +++ b/dotnet/samples/GettingStarted/ModelContextProtocol/Agent_MCP_Server_Auth/Agent_MCP_Server_Auth.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable @@ -15,7 +15,6 @@ - diff --git a/dotnet/samples/GettingStarted/ModelContextProtocol/Agent_MCP_Server_Auth/Program.cs b/dotnet/samples/GettingStarted/ModelContextProtocol/Agent_MCP_Server_Auth/Program.cs index aae520eec9..c9197c573c 100644 --- a/dotnet/samples/GettingStarted/ModelContextProtocol/Agent_MCP_Server_Auth/Program.cs +++ b/dotnet/samples/GettingStarted/ModelContextProtocol/Agent_MCP_Server_Auth/Program.cs @@ -11,7 +11,7 @@ using Microsoft.Agents.AI; using Microsoft.Extensions.Logging; using ModelContextProtocol.Client; -using OpenAI; +using OpenAI.Chat; var endpoint = Environment.GetEnvironmentVariable("AZURE_OPENAI_ENDPOINT") ?? throw new InvalidOperationException("AZURE_OPENAI_ENDPOINT is not set."); var deploymentName = Environment.GetEnvironmentVariable("AZURE_OPENAI_DEPLOYMENT_NAME") ?? "gpt-4o-mini"; diff --git a/dotnet/samples/GettingStarted/ModelContextProtocol/Agent_MCP_Server_Auth/README.md b/dotnet/samples/GettingStarted/ModelContextProtocol/Agent_MCP_Server_Auth/README.md index ae88df95ee..a6505d6524 100644 --- a/dotnet/samples/GettingStarted/ModelContextProtocol/Agent_MCP_Server_Auth/README.md +++ b/dotnet/samples/GettingStarted/ModelContextProtocol/Agent_MCP_Server_Auth/README.md @@ -17,7 +17,7 @@ The sample shows: ## Installing Prerequisites - A self-signed certificate to enable HTTPS use in development, see [dotnet dev-certs](https://learn.microsoft.com/en-us/dotnet/core/tools/dotnet-dev-certs) -- .NET 9.0 or later +- .NET 10.0 or later - A running TestOAuthServer (for OAuth authentication), see [Start the Test OAuth Server](https://github.com/modelcontextprotocol/csharp-sdk/tree/main/samples/ProtectedMcpClient#step-1-start-the-test-oauth-server) - A running ProtectedMCPServer (for MCP services), see [Start the Protected MCP Server](https://github.com/modelcontextprotocol/csharp-sdk/tree/main/samples/ProtectedMcpClient#step-2-start-the-protected-mcp-server) @@ -38,7 +38,7 @@ First, you need to start the TestOAuthServer which provides OAuth authentication ```bash cd \tests\ModelContextProtocol.TestOAuthServer -dotnet run --framework net9.0 +dotnet run --framework net10.0 ``` The OAuth server will start at `https://localhost:7029` diff --git a/dotnet/samples/GettingStarted/ModelContextProtocol/FoundryAgent_Hosted_MCP/FoundryAgent_Hosted_MCP.csproj b/dotnet/samples/GettingStarted/ModelContextProtocol/FoundryAgent_Hosted_MCP/FoundryAgent_Hosted_MCP.csproj index 11c7beb3bf..d40e93232b 100644 --- a/dotnet/samples/GettingStarted/ModelContextProtocol/FoundryAgent_Hosted_MCP/FoundryAgent_Hosted_MCP.csproj +++ b/dotnet/samples/GettingStarted/ModelContextProtocol/FoundryAgent_Hosted_MCP/FoundryAgent_Hosted_MCP.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable diff --git a/dotnet/samples/GettingStarted/ModelContextProtocol/FoundryAgent_Hosted_MCP/Program.cs b/dotnet/samples/GettingStarted/ModelContextProtocol/FoundryAgent_Hosted_MCP/Program.cs index f824f09991..123d666f09 100644 --- a/dotnet/samples/GettingStarted/ModelContextProtocol/FoundryAgent_Hosted_MCP/Program.cs +++ b/dotnet/samples/GettingStarted/ModelContextProtocol/FoundryAgent_Hosted_MCP/Program.cs @@ -34,9 +34,9 @@ options: new() { Name = "MicrosoftLearnAgent", - Instructions = "You answer questions by searching the Microsoft Learn content only.", ChatOptions = new() { + Instructions = "You answer questions by searching the Microsoft Learn content only.", Tools = [mcpTool] }, }); @@ -67,9 +67,9 @@ options: new() { Name = "MicrosoftLearnAgentWithApproval", - Instructions = "You answer questions by searching the Microsoft Learn content only.", ChatOptions = new() { + Instructions = "You answer questions by searching the Microsoft Learn content only.", Tools = [mcpToolWithApproval] }, }); diff --git a/dotnet/samples/GettingStarted/ModelContextProtocol/FoundryAgent_Hosted_MCP/README.md b/dotnet/samples/GettingStarted/ModelContextProtocol/FoundryAgent_Hosted_MCP/README.md index e320a6c3d7..f3be7da576 100644 --- a/dotnet/samples/GettingStarted/ModelContextProtocol/FoundryAgent_Hosted_MCP/README.md +++ b/dotnet/samples/GettingStarted/ModelContextProtocol/FoundryAgent_Hosted_MCP/README.md @@ -2,7 +2,7 @@ Before you begin, ensure you have the following prerequisites: -- .NET 8.0 SDK or later +- .NET 10 SDK or later - Azure Foundry service endpoint and deployment configured - Azure CLI installed and authenticated (for Azure credential authentication) diff --git a/dotnet/samples/GettingStarted/ModelContextProtocol/README.md b/dotnet/samples/GettingStarted/ModelContextProtocol/README.md index 874afa28b8..be1aa83513 100644 --- a/dotnet/samples/GettingStarted/ModelContextProtocol/README.md +++ b/dotnet/samples/GettingStarted/ModelContextProtocol/README.md @@ -6,7 +6,7 @@ The getting started with Model Content Protocol samples demonstrate how to use M Before you begin, ensure you have the following prerequisites: -- .NET 9.0 SDK or later +- .NET 10.0 SDK or later - Azure OpenAI service endpoint and deployment configured - Azure CLI installed and authenticated (for Azure credential authentication) - User has the `Cognitive Services OpenAI Contributor` role for the Azure OpenAI resource. diff --git a/dotnet/samples/GettingStarted/ModelContextProtocol/ResponseAgent_Hosted_MCP/Program.cs b/dotnet/samples/GettingStarted/ModelContextProtocol/ResponseAgent_Hosted_MCP/Program.cs index 19793e64df..ba4249c765 100644 --- a/dotnet/samples/GettingStarted/ModelContextProtocol/ResponseAgent_Hosted_MCP/Program.cs +++ b/dotnet/samples/GettingStarted/ModelContextProtocol/ResponseAgent_Hosted_MCP/Program.cs @@ -8,7 +8,7 @@ using Azure.Identity; using Microsoft.Agents.AI; using Microsoft.Extensions.AI; -using OpenAI; +using OpenAI.Responses; var endpoint = Environment.GetEnvironmentVariable("AZURE_OPENAI_ENDPOINT") ?? throw new InvalidOperationException("AZURE_OPENAI_ENDPOINT is not set."); var deploymentName = Environment.GetEnvironmentVariable("AZURE_OPENAI_DEPLOYMENT_NAME") ?? "gpt-4o-mini"; diff --git a/dotnet/samples/GettingStarted/ModelContextProtocol/ResponseAgent_Hosted_MCP/README.md b/dotnet/samples/GettingStarted/ModelContextProtocol/ResponseAgent_Hosted_MCP/README.md index f84bd8f1b4..c311edae40 100644 --- a/dotnet/samples/GettingStarted/ModelContextProtocol/ResponseAgent_Hosted_MCP/README.md +++ b/dotnet/samples/GettingStarted/ModelContextProtocol/ResponseAgent_Hosted_MCP/README.md @@ -2,7 +2,7 @@ Before you begin, ensure you have the following prerequisites: -- .NET 8.0 SDK or later +- .NET 10 SDK or later - Azure OpenAI service endpoint and deployment configured - Azure CLI installed and authenticated (for Azure credential authentication) - User has the `Cognitive Services OpenAI Contributor` role for the Azure OpenAI resource. diff --git a/dotnet/samples/GettingStarted/ModelContextProtocol/ResponseAgent_Hosted_MCP/ResponseAgent_Hosted_MCP.csproj b/dotnet/samples/GettingStarted/ModelContextProtocol/ResponseAgent_Hosted_MCP/ResponseAgent_Hosted_MCP.csproj index 0eacdab258..41aafe3437 100644 --- a/dotnet/samples/GettingStarted/ModelContextProtocol/ResponseAgent_Hosted_MCP/ResponseAgent_Hosted_MCP.csproj +++ b/dotnet/samples/GettingStarted/ModelContextProtocol/ResponseAgent_Hosted_MCP/ResponseAgent_Hosted_MCP.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable diff --git a/dotnet/samples/GettingStarted/README.md b/dotnet/samples/GettingStarted/README.md index e7249ac33d..7a46d81a62 100644 --- a/dotnet/samples/GettingStarted/README.md +++ b/dotnet/samples/GettingStarted/README.md @@ -8,9 +8,13 @@ of the agent framework. |Sample|Description| |---|---| |[Agents](./Agents/README.md)|Step by step instructions for getting started with agents| +|[Foundry Agents](./FoundryAgents/README.md)|Getting started with Azure Foundry Agents| |[Agent Providers](./AgentProviders/README.md)|Getting started with creating agents using various providers| +|[Agents With Retrieval Augmented Generation (RAG)](./AgentWithRAG/README.md)|Adding Retrieval Augmented Generation (RAG) capabilities to your agents.| +|[Agents With Memory](./AgentWithMemory/README.md)|Adding Memory capabilities to your agents.| |[A2A](./A2A/README.md)|Getting started with A2A (Agent-to-Agent) specific features| |[Agent Open Telemetry](./AgentOpenTelemetry/README.md)|Getting started with OpenTelemetry for agents| |[Agent With OpenAI exchange types](./AgentWithOpenAI/README.md)|Using OpenAI exchange types with agents| +|[Agent With Anthropic](./AgentWithAnthropic/README.md)|Getting started with agents using Anthropic Claude| |[Workflow](./Workflows/README.md)|Getting started with Workflow| |[Model Context Protocol](./ModelContextProtocol/README.md)|Getting started with Model Context Protocol| diff --git a/dotnet/samples/GettingStarted/Workflows/Agents/CustomAgentExecutors/CustomAgentExecutors.csproj b/dotnet/samples/GettingStarted/Workflows/Agents/CustomAgentExecutors/CustomAgentExecutors.csproj index 51b18bdeb2..d0c0656ade 100644 --- a/dotnet/samples/GettingStarted/Workflows/Agents/CustomAgentExecutors/CustomAgentExecutors.csproj +++ b/dotnet/samples/GettingStarted/Workflows/Agents/CustomAgentExecutors/CustomAgentExecutors.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable @@ -16,7 +16,6 @@ - diff --git a/dotnet/samples/GettingStarted/Workflows/Agents/CustomAgentExecutors/Program.cs b/dotnet/samples/GettingStarted/Workflows/Agents/CustomAgentExecutors/Program.cs index 5d5369883c..91f58f460e 100644 --- a/dotnet/samples/GettingStarted/Workflows/Agents/CustomAgentExecutors/Program.cs +++ b/dotnet/samples/GettingStarted/Workflows/Agents/CustomAgentExecutors/Program.cs @@ -118,10 +118,11 @@ internal sealed class SloganWriterExecutor : Executor /// The chat client to use for the AI agent. public SloganWriterExecutor(string id, IChatClient chatClient) : base(id) { - ChatClientAgentOptions agentOptions = new(instructions: "You are a professional slogan writer. You will be given a task to create a slogan.") + ChatClientAgentOptions agentOptions = new() { ChatOptions = new() { + Instructions = "You are a professional slogan writer. You will be given a task to create a slogan.", ResponseFormat = ChatResponseFormat.ForJsonSchema() } }; @@ -193,10 +194,11 @@ internal sealed class FeedbackExecutor : Executor /// The chat client to use for the AI agent. public FeedbackExecutor(string id, IChatClient chatClient) : base(id) { - ChatClientAgentOptions agentOptions = new(instructions: "You are a professional editor. You will be given a slogan and the task it is meant to accomplish.") + ChatClientAgentOptions agentOptions = new() { ChatOptions = new() { + Instructions = "You are a professional editor. You will be given a slogan and the task it is meant to accomplish.", ResponseFormat = ChatResponseFormat.ForJsonSchema() } }; diff --git a/dotnet/samples/GettingStarted/Workflows/Agents/FoundryAgent/FoundryAgent.csproj b/dotnet/samples/GettingStarted/Workflows/Agents/FoundryAgent/FoundryAgent.csproj index 888274205a..f75c7fd28b 100644 --- a/dotnet/samples/GettingStarted/Workflows/Agents/FoundryAgent/FoundryAgent.csproj +++ b/dotnet/samples/GettingStarted/Workflows/Agents/FoundryAgent/FoundryAgent.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable diff --git a/dotnet/samples/GettingStarted/Workflows/Agents/WorkflowAsAnAgent/WorkflowAsAnAgent.csproj b/dotnet/samples/GettingStarted/Workflows/Agents/WorkflowAsAnAgent/WorkflowAsAnAgent.csproj index 51b18bdeb2..d0c0656ade 100644 --- a/dotnet/samples/GettingStarted/Workflows/Agents/WorkflowAsAnAgent/WorkflowAsAnAgent.csproj +++ b/dotnet/samples/GettingStarted/Workflows/Agents/WorkflowAsAnAgent/WorkflowAsAnAgent.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable @@ -16,7 +16,6 @@ - diff --git a/dotnet/samples/GettingStarted/Workflows/Agents/WorkflowAsAnAgent/WorkflowFactory.cs b/dotnet/samples/GettingStarted/Workflows/Agents/WorkflowAsAnAgent/WorkflowFactory.cs index 653ebdf4c2..e418ca7131 100644 --- a/dotnet/samples/GettingStarted/Workflows/Agents/WorkflowAsAnAgent/WorkflowFactory.cs +++ b/dotnet/samples/GettingStarted/Workflows/Agents/WorkflowAsAnAgent/WorkflowFactory.cs @@ -16,7 +16,7 @@ internal static class WorkflowFactory internal static Workflow BuildWorkflow(IChatClient chatClient) { // Create executors - var startExecutor = new ConcurrentStartExecutor(); + var startExecutor = new ChatForwardingExecutor("Start"); var aggregationExecutor = new ConcurrentAggregationExecutor(); AIAgent frenchAgent = GetLanguageAgent("French", chatClient); AIAgent englishAgent = GetLanguageAgent("English", chatClient); @@ -38,33 +38,11 @@ internal static Workflow BuildWorkflow(IChatClient chatClient) private static ChatClientAgent GetLanguageAgent(string targetLanguage, IChatClient chatClient) => new(chatClient, instructions: $"You're a helpful assistant who always responds in {targetLanguage}.", name: $"{targetLanguage}Agent"); - /// - /// Executor that starts the concurrent processing by sending messages to the agents. - /// - private sealed class ConcurrentStartExecutor() : Executor("ConcurrentStartExecutor") - { - protected override RouteBuilder ConfigureRoutes(RouteBuilder routeBuilder) - { - return routeBuilder - .AddHandler>(this.RouteMessages) - .AddHandler(this.RouteTurnTokenAsync); - } - - private ValueTask RouteMessages(List messages, IWorkflowContext context, CancellationToken cancellationToken) - { - return context.SendMessageAsync(messages, cancellationToken: cancellationToken); - } - - private ValueTask RouteTurnTokenAsync(TurnToken token, IWorkflowContext context, CancellationToken cancellationToken) - { - return context.SendMessageAsync(token, cancellationToken: cancellationToken); - } - } - /// /// Executor that aggregates the results from the concurrent agents. /// - private sealed class ConcurrentAggregationExecutor() : Executor>("ConcurrentAggregationExecutor") + private sealed class ConcurrentAggregationExecutor() : + Executor>("ConcurrentAggregationExecutor"), IResettableExecutor { private readonly List _messages = []; @@ -85,5 +63,12 @@ public override async ValueTask HandleAsync(List message, IWorkflow await context.YieldOutputAsync(formattedMessages, cancellationToken); } } + + /// + public ValueTask ResetAsync() + { + this._messages.Clear(); + return default; + } } } diff --git a/dotnet/samples/GettingStarted/Workflows/Checkpoint/CheckpointAndRehydrate/CheckpointAndRehydrate.csproj b/dotnet/samples/GettingStarted/Workflows/Checkpoint/CheckpointAndRehydrate/CheckpointAndRehydrate.csproj index 0a0945caff..2f41070759 100644 --- a/dotnet/samples/GettingStarted/Workflows/Checkpoint/CheckpointAndRehydrate/CheckpointAndRehydrate.csproj +++ b/dotnet/samples/GettingStarted/Workflows/Checkpoint/CheckpointAndRehydrate/CheckpointAndRehydrate.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable diff --git a/dotnet/samples/GettingStarted/Workflows/Checkpoint/CheckpointAndResume/CheckpointAndResume.csproj b/dotnet/samples/GettingStarted/Workflows/Checkpoint/CheckpointAndResume/CheckpointAndResume.csproj index 0a0945caff..2f41070759 100644 --- a/dotnet/samples/GettingStarted/Workflows/Checkpoint/CheckpointAndResume/CheckpointAndResume.csproj +++ b/dotnet/samples/GettingStarted/Workflows/Checkpoint/CheckpointAndResume/CheckpointAndResume.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable diff --git a/dotnet/samples/GettingStarted/Workflows/Checkpoint/CheckpointWithHumanInTheLoop/CheckpointWithHumanInTheLoop.csproj b/dotnet/samples/GettingStarted/Workflows/Checkpoint/CheckpointWithHumanInTheLoop/CheckpointWithHumanInTheLoop.csproj index 0a0945caff..2f41070759 100644 --- a/dotnet/samples/GettingStarted/Workflows/Checkpoint/CheckpointWithHumanInTheLoop/CheckpointWithHumanInTheLoop.csproj +++ b/dotnet/samples/GettingStarted/Workflows/Checkpoint/CheckpointWithHumanInTheLoop/CheckpointWithHumanInTheLoop.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable diff --git a/dotnet/samples/GettingStarted/Workflows/Concurrent/Concurrent/Concurrent.csproj b/dotnet/samples/GettingStarted/Workflows/Concurrent/Concurrent/Concurrent.csproj index 3f3fe6d56c..e756a0bf1d 100644 --- a/dotnet/samples/GettingStarted/Workflows/Concurrent/Concurrent/Concurrent.csproj +++ b/dotnet/samples/GettingStarted/Workflows/Concurrent/Concurrent/Concurrent.csproj @@ -1,8 +1,8 @@ - + Exe - net9.0 + net10.0 enable enable @@ -15,7 +15,6 @@ - diff --git a/dotnet/samples/GettingStarted/Workflows/Concurrent/MapReduce/MapReduce.csproj b/dotnet/samples/GettingStarted/Workflows/Concurrent/MapReduce/MapReduce.csproj index 7282e3fde4..fd311b7be3 100644 --- a/dotnet/samples/GettingStarted/Workflows/Concurrent/MapReduce/MapReduce.csproj +++ b/dotnet/samples/GettingStarted/Workflows/Concurrent/MapReduce/MapReduce.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable diff --git a/dotnet/samples/GettingStarted/Workflows/ConditionalEdges/01_EdgeCondition/01_EdgeCondition.csproj b/dotnet/samples/GettingStarted/Workflows/ConditionalEdges/01_EdgeCondition/01_EdgeCondition.csproj index 17b1cb882a..422c1ca55f 100644 --- a/dotnet/samples/GettingStarted/Workflows/ConditionalEdges/01_EdgeCondition/01_EdgeCondition.csproj +++ b/dotnet/samples/GettingStarted/Workflows/ConditionalEdges/01_EdgeCondition/01_EdgeCondition.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable @@ -16,7 +16,6 @@ - diff --git a/dotnet/samples/GettingStarted/Workflows/ConditionalEdges/01_EdgeCondition/Program.cs b/dotnet/samples/GettingStarted/Workflows/ConditionalEdges/01_EdgeCondition/Program.cs index b6e3d4d513..0f762ea40d 100644 --- a/dotnet/samples/GettingStarted/Workflows/ConditionalEdges/01_EdgeCondition/Program.cs +++ b/dotnet/samples/GettingStarted/Workflows/ConditionalEdges/01_EdgeCondition/Program.cs @@ -85,10 +85,11 @@ private static async Task Main() /// /// A ChatClientAgent configured for spam detection private static ChatClientAgent GetSpamDetectionAgent(IChatClient chatClient) => - new(chatClient, new ChatClientAgentOptions(instructions: "You are a spam detection assistant that identifies spam emails.") + new(chatClient, new ChatClientAgentOptions() { ChatOptions = new() { + Instructions = "You are a spam detection assistant that identifies spam emails.", ResponseFormat = ChatResponseFormat.ForJsonSchema() } }); @@ -98,10 +99,11 @@ private static ChatClientAgent GetSpamDetectionAgent(IChatClient chatClient) => /// /// A ChatClientAgent configured for email assistance private static ChatClientAgent GetEmailAssistantAgent(IChatClient chatClient) => - new(chatClient, new ChatClientAgentOptions(instructions: "You are an email assistant that helps users draft responses to emails with professionalism.") + new(chatClient, new ChatClientAgentOptions() { ChatOptions = new() { + Instructions = "You are an email assistant that helps users draft responses to emails with professionalism.", ResponseFormat = ChatResponseFormat.ForJsonSchema() } }); diff --git a/dotnet/samples/GettingStarted/Workflows/ConditionalEdges/02_SwitchCase/02_SwitchCase.csproj b/dotnet/samples/GettingStarted/Workflows/ConditionalEdges/02_SwitchCase/02_SwitchCase.csproj index 17b1cb882a..422c1ca55f 100644 --- a/dotnet/samples/GettingStarted/Workflows/ConditionalEdges/02_SwitchCase/02_SwitchCase.csproj +++ b/dotnet/samples/GettingStarted/Workflows/ConditionalEdges/02_SwitchCase/02_SwitchCase.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable @@ -16,7 +16,6 @@ - diff --git a/dotnet/samples/GettingStarted/Workflows/ConditionalEdges/02_SwitchCase/Program.cs b/dotnet/samples/GettingStarted/Workflows/ConditionalEdges/02_SwitchCase/Program.cs index 13f0a75bc2..ccda3fa19e 100644 --- a/dotnet/samples/GettingStarted/Workflows/ConditionalEdges/02_SwitchCase/Program.cs +++ b/dotnet/samples/GettingStarted/Workflows/ConditionalEdges/02_SwitchCase/Program.cs @@ -100,10 +100,11 @@ private static async Task Main() /// /// A ChatClientAgent configured for spam detection private static ChatClientAgent GetSpamDetectionAgent(IChatClient chatClient) => - new(chatClient, new ChatClientAgentOptions(instructions: "You are a spam detection assistant that identifies spam emails. Be less confident in your assessments.") + new(chatClient, new ChatClientAgentOptions() { ChatOptions = new() { + Instructions = "You are a spam detection assistant that identifies spam emails. Be less confident in your assessments.", ResponseFormat = ChatResponseFormat.ForJsonSchema() } }); @@ -113,10 +114,11 @@ private static ChatClientAgent GetSpamDetectionAgent(IChatClient chatClient) => /// /// A ChatClientAgent configured for email assistance private static ChatClientAgent GetEmailAssistantAgent(IChatClient chatClient) => - new(chatClient, new ChatClientAgentOptions(instructions: "You are an email assistant that helps users draft responses to emails with professionalism.") + new(chatClient, new ChatClientAgentOptions() { ChatOptions = new() { + Instructions = "You are an email assistant that helps users draft responses to emails with professionalism.", ResponseFormat = ChatResponseFormat.ForJsonSchema() } }); diff --git a/dotnet/samples/GettingStarted/Workflows/ConditionalEdges/03_MultiSelection/03_MultiSelection.csproj b/dotnet/samples/GettingStarted/Workflows/ConditionalEdges/03_MultiSelection/03_MultiSelection.csproj index 17b1cb882a..422c1ca55f 100644 --- a/dotnet/samples/GettingStarted/Workflows/ConditionalEdges/03_MultiSelection/03_MultiSelection.csproj +++ b/dotnet/samples/GettingStarted/Workflows/ConditionalEdges/03_MultiSelection/03_MultiSelection.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable @@ -16,7 +16,6 @@ - diff --git a/dotnet/samples/GettingStarted/Workflows/ConditionalEdges/03_MultiSelection/Program.cs b/dotnet/samples/GettingStarted/Workflows/ConditionalEdges/03_MultiSelection/Program.cs index 9d340cbae3..49faff39da 100644 --- a/dotnet/samples/GettingStarted/Workflows/ConditionalEdges/03_MultiSelection/Program.cs +++ b/dotnet/samples/GettingStarted/Workflows/ConditionalEdges/03_MultiSelection/Program.cs @@ -140,10 +140,11 @@ private static async Task Main() /// /// A ChatClientAgent configured for email analysis private static ChatClientAgent GetEmailAnalysisAgent(IChatClient chatClient) => - new(chatClient, new ChatClientAgentOptions(instructions: "You are a spam detection assistant that identifies spam emails.") + new(chatClient, new ChatClientAgentOptions() { ChatOptions = new() { + Instructions = "You are a spam detection assistant that identifies spam emails.", ResponseFormat = ChatResponseFormat.ForJsonSchema() } }); @@ -153,10 +154,11 @@ private static ChatClientAgent GetEmailAnalysisAgent(IChatClient chatClient) => /// /// A ChatClientAgent configured for email assistance private static ChatClientAgent GetEmailAssistantAgent(IChatClient chatClient) => - new(chatClient, new ChatClientAgentOptions(instructions: "You are an email assistant that helps users draft responses to emails with professionalism.") + new(chatClient, new ChatClientAgentOptions() { ChatOptions = new() { + Instructions = "You are an email assistant that helps users draft responses to emails with professionalism.", ResponseFormat = ChatResponseFormat.ForJsonSchema() } }); @@ -166,10 +168,11 @@ private static ChatClientAgent GetEmailAssistantAgent(IChatClient chatClient) => /// /// A ChatClientAgent configured for email summarization private static ChatClientAgent GetEmailSummaryAgent(IChatClient chatClient) => - new(chatClient, new ChatClientAgentOptions(instructions: "You are an assistant that helps users summarize emails.") + new(chatClient, new ChatClientAgentOptions() { ChatOptions = new() { + Instructions = "You are an assistant that helps users summarize emails.", ResponseFormat = ChatResponseFormat.ForJsonSchema() } }); diff --git a/dotnet/samples/GettingStarted/Workflows/Declarative/ConfirmInput/ConfirmInput.csproj b/dotnet/samples/GettingStarted/Workflows/Declarative/ConfirmInput/ConfirmInput.csproj new file mode 100644 index 0000000000..da32d18b99 --- /dev/null +++ b/dotnet/samples/GettingStarted/Workflows/Declarative/ConfirmInput/ConfirmInput.csproj @@ -0,0 +1,38 @@ + + + + Exe + net10.0 + enable + enable + + + + true + true + true + true + + + + + + + + + + + + + + + + + + + + Always + + + + diff --git a/workflow-samples/HumanInLoop.yaml b/dotnet/samples/GettingStarted/Workflows/Declarative/ConfirmInput/ConfirmInput.yaml similarity index 88% rename from workflow-samples/HumanInLoop.yaml rename to dotnet/samples/GettingStarted/Workflows/Declarative/ConfirmInput/ConfirmInput.yaml index 11c63adb5e..339537c74a 100644 --- a/workflow-samples/HumanInLoop.yaml +++ b/dotnet/samples/GettingStarted/Workflows/Declarative/ConfirmInput/ConfirmInput.yaml @@ -1,8 +1,8 @@ # -# This workflow demonstrates a single agent interaction based on user input. +# This workflow demonstrates how to use the Question action +# to request user input and confirm it matches the original input. # -# Any Foundry Agent may be used to provide the response. -# See: ./setup/QuestionAgent.yaml +# Note: This workflow doesn't make use of any agents. # kind: Workflow trigger: @@ -59,6 +59,3 @@ trigger: Confirmed input: {Local.ConfirmedInput} - - - diff --git a/dotnet/samples/GettingStarted/Workflows/Declarative/ConfirmInput/Program.cs b/dotnet/samples/GettingStarted/Workflows/Declarative/ConfirmInput/Program.cs new file mode 100644 index 0000000000..0e409aa0a0 --- /dev/null +++ b/dotnet/samples/GettingStarted/Workflows/Declarative/ConfirmInput/Program.cs @@ -0,0 +1,39 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Microsoft.Extensions.Configuration; +using Shared.Workflows; + +namespace Demo.Workflows.Declarative.ConfirmInput; + +/// +/// Demonstrate how to use the question action to request user input +/// and confirm it matches the original input. +/// +/// +/// See the README.md file in the parent folder (../README.md) for detailed +/// information about the configuration required to run this sample. +/// +internal sealed class Program +{ + public static async Task Main(string[] args) + { + // Initialize configuration + IConfiguration configuration = Application.InitializeConfig(); + Uri foundryEndpoint = new(configuration.GetValue(Application.Settings.FoundryEndpoint)); + + // Get input from command line or console + string workflowInput = Application.GetInput(args); + + // Create the workflow factory. This class demonstrates how to initialize a + // declarative workflow from a YAML file. Once the workflow is created, it + // can be executed just like any regular workflow. + WorkflowFactory workflowFactory = new("ConfirmInput.yaml", foundryEndpoint); + + // Execute the workflow: The WorkflowRunner demonstrates how to execute + // a workflow, handle the workflow events, and providing external input. + // This also includes the ability to checkpoint workflow state and how to + // resume execution. + WorkflowRunner runner = new(); + await runner.ExecuteAsync(workflowFactory.CreateWorkflow, workflowInput); + } +} diff --git a/dotnet/samples/GettingStarted/Workflows/Declarative/CustomerSupport/CustomerSupport.csproj b/dotnet/samples/GettingStarted/Workflows/Declarative/CustomerSupport/CustomerSupport.csproj new file mode 100644 index 0000000000..583dbc6e8f --- /dev/null +++ b/dotnet/samples/GettingStarted/Workflows/Declarative/CustomerSupport/CustomerSupport.csproj @@ -0,0 +1,38 @@ + + + + Exe + net10.0 + enable + enable + + + + true + true + true + true + + + + + + + + + + + + + + + + + + + + Always + + + + \ No newline at end of file diff --git a/dotnet/samples/GettingStarted/Workflows/Declarative/CustomerSupport/Program.cs b/dotnet/samples/GettingStarted/Workflows/Declarative/CustomerSupport/Program.cs new file mode 100644 index 0000000000..f18b8b4658 --- /dev/null +++ b/dotnet/samples/GettingStarted/Workflows/Declarative/CustomerSupport/Program.cs @@ -0,0 +1,441 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Azure.AI.Projects; +using Azure.AI.Projects.OpenAI; +using Azure.Identity; +using Microsoft.Extensions.AI; +using Microsoft.Extensions.Configuration; +using OpenAI.Responses; +using Shared.Foundry; +using Shared.Workflows; + +namespace Demo.Workflows.Declarative.CustomerSupport; + +/// +/// This workflow demonstrates using multiple agents to provide automated +/// troubleshooting steps to resolve common issues with escalation options. +/// +/// +/// See the README.md file in the parent folder (../README.md) for detailed +/// information about the configuration required to run this sample. +/// +internal sealed class Program +{ + public static async Task Main(string[] args) + { + // Initialize configuration + IConfiguration configuration = Application.InitializeConfig(); + Uri foundryEndpoint = new(configuration.GetValue(Application.Settings.FoundryEndpoint)); + + // Create the ticketing plugin (mock functionality) + TicketingPlugin plugin = new(); + + // Ensure sample agents exist in Foundry. + await CreateAgentsAsync(foundryEndpoint, configuration, plugin); + + // Get input from command line or console + string workflowInput = Application.GetInput(args); + + // Create the workflow factory. This class demonstrates how to initialize a + // declarative workflow from a YAML file. Once the workflow is created, it + // can be executed just like any regular workflow. + WorkflowFactory workflowFactory = + new("CustomerSupport.yaml", foundryEndpoint) + { + Functions = + [ + AIFunctionFactory.Create(plugin.CreateTicket), + AIFunctionFactory.Create(plugin.GetTicket), + AIFunctionFactory.Create(plugin.ResolveTicket), + AIFunctionFactory.Create(plugin.SendNotification), + ] + }; + + // Execute the workflow: The WorkflowRunner demonstrates how to execute + // a workflow, handle the workflow events, and providing external input. + // This also includes the ability to checkpoint workflow state and how to + // resume execution. + WorkflowRunner runner = new(); + await runner.ExecuteAsync(workflowFactory.CreateWorkflow, workflowInput); + } + + private static async Task CreateAgentsAsync(Uri foundryEndpoint, IConfiguration configuration, TicketingPlugin plugin) + { + AIProjectClient aiProjectClient = new(foundryEndpoint, new AzureCliCredential()); + + await aiProjectClient.CreateAgentAsync( + agentName: "SelfServiceAgent", + agentDefinition: DefineSelfServiceAgent(configuration), + agentDescription: "Service agent for CustomerSupport workflow"); + + await aiProjectClient.CreateAgentAsync( + agentName: "TicketingAgent", + agentDefinition: DefineTicketingAgent(configuration, plugin), + agentDescription: "Ticketing agent for CustomerSupport workflow"); + + await aiProjectClient.CreateAgentAsync( + agentName: "TicketRoutingAgent", + agentDefinition: DefineTicketRoutingAgent(configuration, plugin), + agentDescription: "Routing agent for CustomerSupport workflow"); + + await aiProjectClient.CreateAgentAsync( + agentName: "WindowsSupportAgent", + agentDefinition: DefineWindowsSupportAgent(configuration, plugin), + agentDescription: "Windows support agent for CustomerSupport workflow"); + + await aiProjectClient.CreateAgentAsync( + agentName: "TicketResolutionAgent", + agentDefinition: DefineResolutionAgent(configuration, plugin), + agentDescription: "Resolution agent for CustomerSupport workflow"); + + await aiProjectClient.CreateAgentAsync( + agentName: "TicketEscalationAgent", + agentDefinition: TicketEscalationAgent(configuration, plugin), + agentDescription: "Escalate agent for human support"); + } + + private static PromptAgentDefinition DefineSelfServiceAgent(IConfiguration configuration) => + new(configuration.GetValue(Application.Settings.FoundryModelMini)) + { + Instructions = + """ + Use your knowledge to work with the user to provide the best possible troubleshooting steps. + + - If the user confirms that the issue is resolved, then the issue is resolved. + - If the user reports that the issue persists, then escalate. + """, + TextOptions = + new ResponseTextOptions + { + TextFormat = + ResponseTextFormat.CreateJsonSchemaFormat( + "TaskEvaluation", + BinaryData.FromString( + """ + { + "type": "object", + "properties": { + "IsResolved": { + "type": "boolean", + "description": "True if the user issue/ask has been resolved." + }, + "NeedsTicket": { + "type": "boolean", + "description": "True if the user issue/ask requires that a ticket be filed." + }, + "IssueDescription": { + "type": "string", + "description": "A concise description of the issue." + }, + "AttemptedResolutionSteps": { + "type": "string", + "description": "An outline of the steps taken to attempt resolution." + } + }, + "required": ["IsResolved", "NeedsTicket", "IssueDescription", "AttemptedResolutionSteps"], + "additionalProperties": false + } + """), + jsonSchemaFormatDescription: null, + jsonSchemaIsStrict: true), + } + }; + + private static PromptAgentDefinition DefineTicketingAgent(IConfiguration configuration, TicketingPlugin plugin) => + new(configuration.GetValue(Application.Settings.FoundryModelMini)) + { + Instructions = + """ + Always create a ticket in Azure DevOps using the available tools. + + Include the following information in the TicketSummary. + + - Issue description: {{IssueDescription}} + - Attempted resolution steps: {{AttemptedResolutionSteps}} + + After creating the ticket, provide the user with the ticket ID. + """, + Tools = + { + AIFunctionFactory.Create(plugin.CreateTicket).AsOpenAIResponseTool() + }, + StructuredInputs = + { + ["IssueDescription"] = + new StructuredInputDefinition + { + IsRequired = false, + DefaultValue = BinaryData.FromString(@"""unknown"""), + Description = "A concise description of the issue.", + }, + ["AttemptedResolutionSteps"] = + new StructuredInputDefinition + { + IsRequired = false, + DefaultValue = BinaryData.FromString(@"""unknown"""), + Description = "An outline of the steps taken to attempt resolution.", + } + }, + TextOptions = + new ResponseTextOptions + { + TextFormat = + ResponseTextFormat.CreateJsonSchemaFormat( + "TaskEvaluation", + BinaryData.FromString( + """ + { + "type": "object", + "properties": { + "TicketId": { + "type": "string", + "description": "The identifier of the ticket created in response to the user issue." + }, + "TicketSummary": { + "type": "string", + "description": "The summary of the ticket created in response to the user issue." + } + }, + "required": ["TicketId", "TicketSummary"], + "additionalProperties": false + } + """), + jsonSchemaFormatDescription: null, + jsonSchemaIsStrict: true), + } + }; + + private static PromptAgentDefinition DefineTicketRoutingAgent(IConfiguration configuration, TicketingPlugin plugin) => + new(configuration.GetValue(Application.Settings.FoundryModelMini)) + { + Instructions = + """ + Determine how to route the given issue to the appropriate support team. + + Choose from the available teams and their functions: + - Windows Activation Support: Windows license activation issues + - Windows Support: Windows related issues + - Azure Support: Azure related issues + - Network Support: Network related issues + - Hardware Support: Hardware related issues + - Microsoft Office Support: Microsoft Office related issues + - General Support: General issues not related to the above categories + """, + Tools = + { + AIFunctionFactory.Create(plugin.GetTicket).AsOpenAIResponseTool(), + }, + TextOptions = + new ResponseTextOptions + { + TextFormat = + ResponseTextFormat.CreateJsonSchemaFormat( + "TaskEvaluation", + BinaryData.FromString( + """ + { + "type": "object", + "properties": { + "TeamName": { + "type": "string", + "description": "The name of the team to route the issue" + } + }, + "required": ["TeamName"], + "additionalProperties": false + } + """), + jsonSchemaFormatDescription: null, + jsonSchemaIsStrict: true), + } + }; + + private static PromptAgentDefinition DefineWindowsSupportAgent(IConfiguration configuration, TicketingPlugin plugin) => + new(configuration.GetValue(Application.Settings.FoundryModelMini)) + { + Instructions = + """ + Use your knowledge to work with the user to provide the best possible troubleshooting steps + for issues related to Windows operating system. + + - Utilize the "Attempted Resolutions Steps" as a starting point for your troubleshooting. + - Never escalate without troubleshooting with the user. + - If the user confirms that the issue is resolved, then the issue is resolved. + - If the user reports that the issue persists, then escalate. + + Issue: {{IssueDescription}} + Attempted Resolution Steps: {{AttemptedResolutionSteps}} + """, + StructuredInputs = + { + ["IssueDescription"] = + new StructuredInputDefinition + { + IsRequired = false, + DefaultValue = BinaryData.FromString(@"""unknown"""), + Description = "A concise description of the issue.", + }, + ["AttemptedResolutionSteps"] = + new StructuredInputDefinition + { + IsRequired = false, + DefaultValue = BinaryData.FromString(@"""unknown"""), + Description = "An outline of the steps taken to attempt resolution.", + } + }, + Tools = + { + AIFunctionFactory.Create(plugin.GetTicket).AsOpenAIResponseTool(), + }, + TextOptions = + new ResponseTextOptions + { + TextFormat = + ResponseTextFormat.CreateJsonSchemaFormat( + "TaskEvaluation", + BinaryData.FromString( + """ + { + "type": "object", + "properties": { + "IsResolved": { + "type": "boolean", + "description": "True if the user issue/ask has been resolved." + }, + "NeedsEscalation": { + "type": "boolean", + "description": "True resolution could not be achieved and the issue/ask requires escalation." + }, + "ResolutionSummary": { + "type": "string", + "description": "The summary of the steps that led to resolution." + } + }, + "required": ["IsResolved", "NeedsEscalation", "ResolutionSummary"], + "additionalProperties": false + } + """), + jsonSchemaFormatDescription: null, + jsonSchemaIsStrict: true), + } + }; + + private static PromptAgentDefinition DefineResolutionAgent(IConfiguration configuration, TicketingPlugin plugin) => + new(configuration.GetValue(Application.Settings.FoundryModelMini)) + { + Instructions = + """ + Resolve the following ticket in Azure DevOps. + Always include the resolution details. + + - Ticket ID: #{{TicketId}} + - Resolution Summary: {{ResolutionSummary}} + """, + Tools = + { + AIFunctionFactory.Create(plugin.ResolveTicket).AsOpenAIResponseTool(), + }, + StructuredInputs = + { + ["TicketId"] = + new StructuredInputDefinition + { + IsRequired = false, + DefaultValue = BinaryData.FromString(@"""unknown"""), + Description = "The identifier of the ticket being resolved.", + }, + ["ResolutionSummary"] = + new StructuredInputDefinition + { + IsRequired = false, + DefaultValue = BinaryData.FromString(@"""unknown"""), + Description = "The steps taken to resolve the issue.", + } + } + }; + + private static PromptAgentDefinition TicketEscalationAgent(IConfiguration configuration, TicketingPlugin plugin) => + new(configuration.GetValue(Application.Settings.FoundryModelMini)) + { + Instructions = + """ + You escalate the provided issue to human support team by sending an email if the issue is not resolved. + + Here are some additional details that might help: + - TicketId : {{TicketId}} + - IssueDescription : {{IssueDescription}} + - AttemptedResolutionSteps : {{AttemptedResolutionSteps}} + + Before escalating, gather the user's email address for follow-up. + If not known, ask the user for their email address so that the support team can reach them when needed. + + When sending the email, include the following details: + - To: support@contoso.com + - Cc: user's email address + - Subject of the email: "Support Ticket - {TicketId} - [Compact Issue Description]" + - Body: + - Issue description + - Attempted resolution steps + - User's email address + - Any other relevant information from the conversation history + + Assure the user that their issue will be resolved and provide them with a ticket ID for reference. + """, + Tools = + { + AIFunctionFactory.Create(plugin.GetTicket).AsOpenAIResponseTool(), + AIFunctionFactory.Create(plugin.SendNotification).AsOpenAIResponseTool(), + }, + StructuredInputs = + { + ["TicketId"] = + new StructuredInputDefinition + { + IsRequired = false, + DefaultValue = BinaryData.FromString(@"""unknown"""), + Description = "The identifier of the ticket being escalated.", + }, + ["IssueDescription"] = + new StructuredInputDefinition + { + IsRequired = false, + DefaultValue = BinaryData.FromString(@"""unknown"""), + Description = "A concise description of the issue.", + }, + ["ResolutionSummary"] = + new StructuredInputDefinition + { + IsRequired = false, + DefaultValue = BinaryData.FromString(@"""unknown"""), + Description = "An outline of the steps taken to attempt resolution.", + } + }, + TextOptions = + new ResponseTextOptions + { + TextFormat = + ResponseTextFormat.CreateJsonSchemaFormat( + "TaskEvaluation", + BinaryData.FromString( + """ + { + "type": "object", + "properties": { + "IsComplete": { + "type": "boolean", + "description": "Has the email been sent and no more user input is required." + }, + "UserMessage": { + "type": "string", + "description": "A natural language message to the user." + } + }, + "required": ["IsComplete", "UserMessage"], + "additionalProperties": false + } + """), + jsonSchemaFormatDescription: null, + jsonSchemaIsStrict: true), + } + }; +} diff --git a/dotnet/samples/GettingStarted/Workflows/Declarative/CustomerSupport/TicketingPlugin.cs b/dotnet/samples/GettingStarted/Workflows/Declarative/CustomerSupport/TicketingPlugin.cs new file mode 100644 index 0000000000..831af0c4d6 --- /dev/null +++ b/dotnet/samples/GettingStarted/Workflows/Declarative/CustomerSupport/TicketingPlugin.cs @@ -0,0 +1,85 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.ComponentModel; + +namespace Demo.Workflows.Declarative.CustomerSupport; + +internal sealed class TicketingPlugin +{ + private readonly Dictionary _ticketStore = []; + + [Description("Retrieve a ticket by identifier from Azure DevOps.")] + public TicketItem? GetTicket(string id) + { + Trace(nameof(GetTicket)); + + this._ticketStore.TryGetValue(id, out TicketItem? ticket); + + return ticket; + } + + [Description("Create a ticket in Azure DevOps and return its identifier.")] + public string CreateTicket(string subject, string description, string notes) + { + Trace(nameof(CreateTicket)); + + TicketItem ticket = new() + { + Subject = subject, + Description = description, + Notes = notes, + Id = Guid.NewGuid().ToString("N"), + }; + + this._ticketStore[ticket.Id] = ticket; + + return ticket.Id; + } + + [Description("Resolve an existing ticket in Azure DevOps given its identifier.")] + public void ResolveTicket(string id, string resolutionSummary) + { + Trace(nameof(ResolveTicket)); + + if (this._ticketStore.TryGetValue(id, out TicketItem? ticket)) + { + ticket.Status = TicketStatus.Resolved; + } + } + + [Description("Send an email notification to escalate ticket engagement.")] + public void SendNotification(string id, string email, string cc, string body) + { + Trace(nameof(SendNotification)); + } + + private static void Trace(string functionName) + { + Console.ForegroundColor = ConsoleColor.DarkMagenta; + try + { + Console.WriteLine($"\nFUNCTION: {functionName}"); + } + finally + { + Console.ResetColor(); + } + } + + public enum TicketStatus + { + Open, + InProgress, + Resolved, + Closed, + } + + public sealed class TicketItem + { + public TicketStatus Status { get; set; } = TicketStatus.Open; + public string Subject { get; init; } = string.Empty; + public string Id { get; init; } = string.Empty; + public string Description { get; init; } = string.Empty; + public string Notes { get; init; } = string.Empty; + } +} diff --git a/dotnet/samples/GettingStarted/Workflows/Declarative/DeepResearch/DeepResearch.csproj b/dotnet/samples/GettingStarted/Workflows/Declarative/DeepResearch/DeepResearch.csproj new file mode 100644 index 0000000000..413fa56210 --- /dev/null +++ b/dotnet/samples/GettingStarted/Workflows/Declarative/DeepResearch/DeepResearch.csproj @@ -0,0 +1,41 @@ + + + + Exe + net10.0 + enable + enable + + + + true + true + true + true + + + + + + + + + + + + + + + + + + + + Always + + + Always + + + + diff --git a/dotnet/samples/GettingStarted/Workflows/Declarative/DeepResearch/Program.cs b/dotnet/samples/GettingStarted/Workflows/Declarative/DeepResearch/Program.cs new file mode 100644 index 0000000000..7aaa61b398 --- /dev/null +++ b/dotnet/samples/GettingStarted/Workflows/Declarative/DeepResearch/Program.cs @@ -0,0 +1,281 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Azure.AI.Projects; +using Azure.AI.Projects.OpenAI; +using Azure.Identity; +using Microsoft.Extensions.Configuration; +using OpenAI.Responses; +using Shared.Foundry; +using Shared.Workflows; + +namespace Demo.Workflows.Declarative.DeepResearch; + +/// +/// Demonstrate a declarative workflow that accomplishes a task +/// using the Magentic orchestration pattern developed by AutoGen. +/// +/// +/// See the README.md file in the parent folder (../README.md) for detailed +/// information about the configuration required to run this sample. +/// +internal sealed class Program +{ + public static async Task Main(string[] args) + { + // Initialize configuration + IConfiguration configuration = Application.InitializeConfig(); + Uri foundryEndpoint = new(configuration.GetValue(Application.Settings.FoundryEndpoint)); + + // Ensure sample agents exist in Foundry. + await CreateAgentsAsync(foundryEndpoint, configuration); + + // Get input from command line or console + string workflowInput = Application.GetInput(args); + + // Create the workflow factory. This class demonstrates how to initialize a + // declarative workflow from a YAML file. Once the workflow is created, it + // can be executed just like any regular workflow. + WorkflowFactory workflowFactory = new("DeepResearch.yaml", foundryEndpoint); + + // Execute the workflow: The WorkflowRunner demonstrates how to execute + // a workflow, handle the workflow events, and providing external input. + // This also includes the ability to checkpoint workflow state and how to + // resume execution. + WorkflowRunner runner = new(); + await runner.ExecuteAsync(workflowFactory.CreateWorkflow, workflowInput); + } + + private static async Task CreateAgentsAsync(Uri foundryEndpoint, IConfiguration configuration) + { + AIProjectClient aiProjectClient = new(foundryEndpoint, new AzureCliCredential()); + + await aiProjectClient.CreateAgentAsync( + agentName: "ResearchAgent", + agentDefinition: DefineResearchAgent(configuration), + agentDescription: "Planner agent for DeepResearch workflow"); + + await aiProjectClient.CreateAgentAsync( + agentName: "PlannerAgent", + agentDefinition: DefinePlannerAgent(configuration), + agentDescription: "Planner agent for DeepResearch workflow"); + + await aiProjectClient.CreateAgentAsync( + agentName: "ManagerAgent", + agentDefinition: DefineManagerAgent(configuration), + agentDescription: "Manager agent for DeepResearch workflow"); + + await aiProjectClient.CreateAgentAsync( + agentName: "SummaryAgent", + agentDefinition: DefineSummaryAgent(configuration), + agentDescription: "Summary agent for DeepResearch workflow"); + + await aiProjectClient.CreateAgentAsync( + agentName: "KnowledgeAgent", + agentDefinition: DefineKnowledgeAgent(configuration), + agentDescription: "Research agent for DeepResearch workflow"); + + await aiProjectClient.CreateAgentAsync( + agentName: "CoderAgent", + agentDefinition: DefineCoderAgent(configuration), + agentDescription: "Coder agent for DeepResearch workflow"); + + await aiProjectClient.CreateAgentAsync( + agentName: "WeatherAgent", + agentDefinition: DefineWeatherAgent(configuration), + agentDescription: "Weather agent for DeepResearch workflow"); + } + + private static PromptAgentDefinition DefineResearchAgent(IConfiguration configuration) => + new(configuration.GetValue(Application.Settings.FoundryModelFull)) + { + Instructions = + """ + In order to help begin addressing the user request, please answer the following pre-survey to the best of your ability. + Keep in mind that you are Ken Jennings-level with trivia, and Mensa-level with puzzles, so there should be a deep well to draw from. + + Here is the pre-survey: + + 1. Please list any specific facts or figures that are GIVEN in the request itself. It is possible that there are none. + 2. Please list any facts that may need to be looked up, and WHERE SPECIFICALLY they might be found. In some cases, authoritative sources are mentioned in the request itself. + 3. Please list any facts that may need to be derived (e.g., via logical deduction, simulation, or computation) + 4. Please list any facts that are recalled from memory, hunches, well-reasoned guesses, etc. + + When answering this survey, keep in mind that 'facts' will typically be specific names, dates, statistics, etc. Your answer must only use the headings: + + 1. GIVEN OR VERIFIED FACTS + 2. FACTS TO LOOK UP + 3. FACTS TO DERIVE + 4. EDUCATED GUESSES + + DO NOT include any other headings or sections in your response. DO NOT list next steps or plans until asked to do so. + """, + Tools = + { + //AgentTool.CreateBingGroundingTool( // TODO: Use Bing Grounding when available + // new BingGroundingSearchToolParameters( + // [new BingGroundingSearchConfiguration(this.GetSetting(Settings.FoundryGroundingTool))])) + } + }; + + private static PromptAgentDefinition DefinePlannerAgent(IConfiguration configuration) => + new(configuration.GetValue(Application.Settings.FoundryModelMini)) + { + Instructions = // TODO: Use Structured Inputs / Prompt Template + """ + Your only job is to devise an efficient plan that identifies (by name) how a team member may contribute to addressing the user request. + + Only select the following team which is listed as "- [Name]: [Description]" + + - WeatherAgent: Able to retrieve weather information + - CoderAgent: Able to write and execute Python code + - KnowledgeAgent: Able to perform generic websearches + + The plan must be a bullet point list must be in the form "- [AgentName]: [Specific action or task for that agent to perform]" + + Remember, there is no requirement to involve the entire team -- only select team member's whose particular expertise is required for this task. + """ + }; + + private static PromptAgentDefinition DefineManagerAgent(IConfiguration configuration) => + new(configuration.GetValue(Application.Settings.FoundryModelMini)) + { + Instructions = // TODO: Use Structured Inputs / Prompt Template + """ + Recall we have assembled the following team: + + - KnowledgeAgent: Able to perform generic websearches + - CoderAgent: Able to write and execute Python code + - WeatherAgent: Able to retrieve weather information + + To make progress on the request, please answer the following questions, including necessary reasoning: + - Is the request fully satisfied? (True if complete, or False if the original request has yet to be SUCCESSFULLY and FULLY addressed) + - Are we in a loop where we are repeating the same requests and / or getting the same responses from an agent multiple times? Loops can span multiple turns, and can include repeated actions like scrolling up or down more than a handful of times. + - Are we making forward progress? (True if just starting, or recent messages are adding value. False if recent messages show evidence of being stuck in a loop or if there is evidence of significant barriers to success such as the inability to read from a required file) + - Who should speak next? (select from: KnowledgeAgent, CoderAgent, WeatherAgent) + - What instruction or question would you give this team member? (Phrase as if speaking directly to them, and include any specific information they may need) + """, + TextOptions = + new ResponseTextOptions + { + TextFormat = + ResponseTextFormat.CreateJsonSchemaFormat( + "TaskEvaluation", + BinaryData.FromString( + """ + { + "type": "object", + "properties": { + "is_request_satisfied": { + "type": "object", + "properties": { + "reason": { "type": "string" }, + "answer": { "type": "boolean" } + }, + "required": ["reason", "answer"], + "additionalProperties": false + }, + "is_in_loop": { + "type": "object", + "properties": { + "reason": { "type": "string" }, + "answer": { "type": "boolean" } + }, + "required": ["reason", "answer"], + "additionalProperties": false + }, + "is_progress_being_made": { + "type": "object", + "properties": { + "reason": { "type": "string" }, + "answer": { "type": "boolean" } + }, + "required": ["reason", "answer"], + "additionalProperties": false + }, + "next_speaker": { + "type": "object", + "properties": { + "reason": { "type": "string" }, + "answer": { + "type": "string" + } + }, + "required": ["reason", "answer"], + "additionalProperties": false + }, + "instruction_or_question": { + "type": "object", + "properties": { + "reason": { "type": "string" }, + "answer": { "type": "string" } + }, + "required": ["reason", "answer"], + "additionalProperties": false + } + }, + "required": ["is_request_satisfied", "is_in_loop", "is_progress_being_made", "next_speaker", "instruction_or_question"], + "additionalProperties": false + } + """), + jsonSchemaFormatDescription: null, + jsonSchemaIsStrict: true), + } + }; + + private static PromptAgentDefinition DefineSummaryAgent(IConfiguration configuration) => + new(configuration.GetValue(Application.Settings.FoundryModelMini)) + { + Instructions = + """ + We have completed the task. + + Based only on the conversation and without adding any new information, + synthesize the result of the conversation as a complete response to the user task. + + The user will only ever see this last response and not the entire conversation, + so please ensure it is complete and self-contained. + """ + }; + + private static PromptAgentDefinition DefineKnowledgeAgent(IConfiguration configuration) => + new(configuration.GetValue(Application.Settings.FoundryModelMini)) + { + Tools = + { + //AgentTool.CreateBingGroundingTool( // TODO: Use Bing Grounding when available + // new BingGroundingSearchToolParameters( + // [new BingGroundingSearchConfiguration(this.GetSetting(Settings.FoundryGroundingTool))])) + } + }; + + private static PromptAgentDefinition DefineCoderAgent(IConfiguration configuration) => + new(configuration.GetValue(Application.Settings.FoundryModelMini)) + { + Instructions = + """ + You solve problem by writing and executing code. + """, + Tools = + { + ResponseTool.CreateCodeInterpreterTool( + new(CodeInterpreterToolContainerConfiguration.CreateAutomaticContainerConfiguration())) + } + }; + + private static PromptAgentDefinition DefineWeatherAgent(IConfiguration configuration) => + new(configuration.GetValue(Application.Settings.FoundryModelMini)) + { + Instructions = + """ + You are a weather expert. + """, + Tools = + { + AgentTool.CreateOpenApiTool( + new OpenAPIFunctionDefinition( + "weather-forecast", + BinaryData.FromString(File.ReadAllText(Path.Combine(AppContext.BaseDirectory, "wttr.json"))), + new OpenAPIAnonymousAuthenticationDetails())) + } + }; +} diff --git a/workflow-samples/wttr.json b/dotnet/samples/GettingStarted/Workflows/Declarative/DeepResearch/wttr.json similarity index 100% rename from workflow-samples/wttr.json rename to dotnet/samples/GettingStarted/Workflows/Declarative/DeepResearch/wttr.json diff --git a/dotnet/samples/GettingStarted/Workflows/Declarative/ExecuteCode/ExecuteCode.csproj b/dotnet/samples/GettingStarted/Workflows/Declarative/ExecuteCode/ExecuteCode.csproj index 72afa29cda..9725826c7a 100644 --- a/dotnet/samples/GettingStarted/Workflows/Declarative/ExecuteCode/ExecuteCode.csproj +++ b/dotnet/samples/GettingStarted/Workflows/Declarative/ExecuteCode/ExecuteCode.csproj @@ -2,9 +2,7 @@ Exe - net9.0 - net9.0 - $(ProjectsDebugTargetFrameworks) + net10.0 enable enable 5ee045b0-aea3-4f08-8d31-32d1a6f8fed0 @@ -12,7 +10,9 @@ - true + true + true + true @@ -27,6 +27,7 @@ + diff --git a/dotnet/samples/GettingStarted/Workflows/Declarative/ExecuteCode/Generated.cs b/dotnet/samples/GettingStarted/Workflows/Declarative/ExecuteCode/Generated.cs index d1c8d45082..6fc508064e 100644 --- a/dotnet/samples/GettingStarted/Workflows/Declarative/ExecuteCode/Generated.cs +++ b/dotnet/samples/GettingStarted/Workflows/Declarative/ExecuteCode/Generated.cs @@ -5,6 +5,7 @@ // ------------------------------------------------------------------------------ #nullable enable +#pragma warning disable IDE0005 // Extra using directive is ok. using System; using System.Collections; @@ -18,7 +19,7 @@ using Microsoft.Agents.AI.Workflows.Declarative.Kit; using Microsoft.Extensions.AI; -namespace Test.WorkflowProviders; +namespace Demo.DeclarativeCode; /// /// This class provides a factory method to create a instance. @@ -29,7 +30,7 @@ namespace Test.WorkflowProviders; /// To learn more about Power FX, see: /// https://learn.microsoft.com/power-platform/power-fx/formula-reference-copilot-studio /// -public static class TestWorkflowProvider +public static class SampleWorkflowProvider { /// /// The root executor for a declarative workflow. @@ -42,408 +43,27 @@ internal sealed class WorkflowDemoRootExecutor( { protected override async ValueTask ExecuteAsync(TInput message, IWorkflowContext context, CancellationToken cancellationToken) { - // Set environment variables - await this.InitializeEnvironmentAsync( - context, - "FOUNDRY_AGENT_RESEARCHWEB", - "FOUNDRY_AGENT_RESEARCHANALYST", - "FOUNDRY_AGENT_RESEARCHCODER", - "FOUNDRY_AGENT_RESEARCHMANAGER", - "FOUNDRY_AGENT_RESEARCHWEATHER").ConfigureAwait(false); - - // Initialize variables - await context.QueueStateUpdateAsync("AgentResponse", UnassignedValue.Instance, "Local"); - await context.QueueStateUpdateAsync("AgentResponseText", UnassignedValue.Instance, "Local"); - await context.QueueStateUpdateAsync("AvailableAgents", UnassignedValue.Instance, "Local"); - await context.QueueStateUpdateAsync("FinalResponse", UnassignedValue.Instance, "Local"); - await context.QueueStateUpdateAsync("InputTask", UnassignedValue.Instance, "Local"); - await context.QueueStateUpdateAsync("InternalConversationId", UnassignedValue.Instance, "Local"); - await context.QueueStateUpdateAsync("NextSpeaker", UnassignedValue.Instance, "Local"); - await context.QueueStateUpdateAsync("Plan", UnassignedValue.Instance, "Local"); - await context.QueueStateUpdateAsync("ProgressLedgerUpdate", UnassignedValue.Instance, "Local"); - await context.QueueStateUpdateAsync("RestartCount", UnassignedValue.Instance, "Local"); - await context.QueueStateUpdateAsync("SeedTask", UnassignedValue.Instance, "Local"); - await context.QueueStateUpdateAsync("StallCount", UnassignedValue.Instance, "Local"); - await context.QueueStateUpdateAsync("TaskFacts", UnassignedValue.Instance, "Local"); - await context.QueueStateUpdateAsync("TaskInstructions", UnassignedValue.Instance, "Local"); - await context.QueueStateUpdateAsync("TeamDescription", UnassignedValue.Instance, "Local"); - await context.QueueStateUpdateAsync("TypedProgressLedger", UnassignedValue.Instance, "Local"); - } - } - - /// - /// Assigns an evaluated expression, other variable, or literal value to the "Local.AvailableAgents" variable. - /// - internal sealed class SetvariableAaslmfExecutor(FormulaSession session) : ActionExecutor(id: "setVariable_aASlmF", session) - { - // - protected override async ValueTask ExecuteAsync(IWorkflowContext context, CancellationToken cancellationToken) - { - object? evaluatedValue = await context.EvaluateValueAsync(""" - [ - { - name: "WeatherAgent", - description: "Able to retrieve weather information", - agentid: Env.FOUNDRY_AGENT_RESEARCHWEATHER - }, - { - name: "CoderAgent", - description: "Able to write and execute Python code", - agentid: Env.FOUNDRY_AGENT_RESEARCHCODER - }, - { - name: "WebAgent", - description: "Able to perform generic websearches", - agentid: Env.FOUNDRY_AGENT_RESEARCHWEB - } - ] - """); - await context.QueueStateUpdateAsync(key: "AvailableAgents", value: evaluatedValue, scopeName: "Local"); - - return default; - } - } - - /// - /// Assigns an evaluated expression, other variable, or literal value to the "Local.TeamDescription" variable. - /// - internal sealed class SetvariableV6yeboExecutor(FormulaSession session) : ActionExecutor(id: "setVariable_V6yEbo", session) - { - // - protected override async ValueTask ExecuteAsync(IWorkflowContext context, CancellationToken cancellationToken) - { - object? evaluatedValue = await context.EvaluateValueAsync(""" - Concat(ForAll(Local.AvailableAgents, $"- " & name & $": " & description), Value, " - ") - """); - await context.QueueStateUpdateAsync(key: "TeamDescription", value: evaluatedValue, scopeName: "Local"); - - return default; - } - } - - /// - /// Assigns an evaluated expression, other variable, or literal value to the "Local.InputTask" variable. - /// - internal sealed class SetvariableNz2u0lExecutor(FormulaSession session) : ActionExecutor(id: "setVariable_NZ2u0l", session) - { - // - protected override async ValueTask ExecuteAsync(IWorkflowContext context, CancellationToken cancellationToken) - { - object? evaluatedValue = await context.EvaluateValueAsync("System.LastMessage.Text"); - await context.QueueStateUpdateAsync(key: "InputTask", value: evaluatedValue, scopeName: "Local"); - - return default; - } - } - - /// - /// Assigns an evaluated expression, other variable, or literal value to the "Local.SeedTask" variable. - /// - internal sealed class Setvariable10U2znExecutor(FormulaSession session) : ActionExecutor(id: "setVariable_10u2ZN", session) - { - // - protected override async ValueTask ExecuteAsync(IWorkflowContext context, CancellationToken cancellationToken) - { - object? evaluatedValue = await context.EvaluateValueAsync("UserMessage(Local.InputTask)"); - await context.QueueStateUpdateAsync(key: "SeedTask", value: evaluatedValue, scopeName: "Local"); - - return default; - } - } - - /// - /// Formats a message template and sends an activity event. - /// - internal sealed class SendactivityYfsbryExecutor(FormulaSession session) : ActionExecutor(id: "sendActivity_yFsbRy", session) - { - // - protected override async ValueTask ExecuteAsync(IWorkflowContext context, CancellationToken cancellationToken) - { - string activityText = - await context.FormatTemplateAsync( - """ - Analyzing facts... - """ - ); - AgentRunResponse response = new([new ChatMessage(ChatRole.Assistant, activityText)]); - await context.AddEventAsync(new AgentRunResponseEvent(this.Id, response)); - - return default; - } - } - - /// - /// Creates a new conversation and stores the identifier value to the "Local.InternalConversationId" variable. - /// - internal sealed class Conversation1A2b3cExecutor(FormulaSession session, WorkflowAgentProvider agentProvider) : ActionExecutor(id: "conversation_1a2b3c", session) - { - protected override async ValueTask ExecuteAsync(IWorkflowContext context, CancellationToken cancellationToken) - { - string conversationId = await agentProvider.CreateConversationAsync(cancellationToken); - await context.QueueStateUpdateAsync(key: "InternalConversationId", value: conversationId, scopeName: "Local"); - - return default; - } - } - - /// - /// Invokes an agent to process messages and return a response within a conversation context. - /// - internal sealed class QuestionUdomuwExecutor(FormulaSession session, WorkflowAgentProvider agentProvider) : AgentExecutor(id: "question_UDoMUw", session, agentProvider) - { - // - protected override async ValueTask ExecuteAsync(IWorkflowContext context, CancellationToken cancellationToken) - { - string? agentName = await context.ReadStateAsync(key: "FOUNDRY_AGENT_RESEARCHANALYST", scopeName: "Env"); - - if (string.IsNullOrWhiteSpace(agentName)) - { - throw new InvalidOperationException($"Agent name must be defined: {this.Id}"); - } - - string? conversationId = await context.ReadStateAsync(key: "InternalConversationId", scopeName: "Local"); - bool autoSend = true; - string additionalInstructions = - await context.FormatTemplateAsync( - """ - In order to help begin addressing the user request, please answer the following pre-survey to the best of your ability. - Keep in mind that you are Ken Jennings-level with trivia, and Mensa-level with puzzles, so there should be a deep well to draw from. - - Here is the pre-survey: - - 1. Please list any specific facts or figures that are GIVEN in the request itself. It is possible that there are none. - 2. Please list any facts that may need to be looked up, and WHERE SPECIFICALLY they might be found. In some cases, authoritative sources are mentioned in the request itself. - 3. Please list any facts that may need to be derived (e.g., via logical deduction, simulation, or computation) - 4. Please list any facts that are recalled from memory, hunches, well-reasoned guesses, etc. - - When answering this survey, keep in mind that 'facts' will typically be specific names, dates, statistics, etc. Your answer must only use the headings: - - 1. GIVEN OR VERIFIED FACTS - 2. FACTS TO LOOK UP - 3. FACTS TO DERIVE - 4. EDUCATED GUESSES - - DO NOT include any other headings or sections in your response. DO NOT list next steps or plans until asked to do so. - """); - IList? inputMessages = await context.EvaluateListAsync("UserMessage(Local.InputTask)"); - - AgentRunResponse agentResponse = - await InvokeAgentAsync( - context, - agentName, - conversationId, - autoSend, - additionalInstructions, - inputMessages, - cancellationToken); - - if (autoSend) - { - await context.AddEventAsync(new AgentRunResponseEvent(this.Id, agentResponse)); - } - - await context.QueueStateUpdateAsync(key: "TaskFacts", value: agentResponse.Messages, scopeName: "Local"); - - return default; - } - } - - /// - /// Formats a message template and sends an activity event. - /// - internal sealed class SendactivityYfsbrzExecutor(FormulaSession session) : ActionExecutor(id: "sendActivity_yFsbRz", session) - { - // - protected override async ValueTask ExecuteAsync(IWorkflowContext context, CancellationToken cancellationToken) - { - string activityText = - await context.FormatTemplateAsync( - """ - Creating a plan... - """ - ); - AgentRunResponse response = new([new ChatMessage(ChatRole.Assistant, activityText)]); - await context.AddEventAsync(new AgentRunResponseEvent(this.Id, response)); - - return default; } } /// /// Invokes an agent to process messages and return a response within a conversation context. /// - internal sealed class QuestionDsbajuExecutor(FormulaSession session, WorkflowAgentProvider agentProvider) : AgentExecutor(id: "question_DsBaJU", session, agentProvider) + internal sealed class QuestionStudentExecutor(FormulaSession session, WorkflowAgentProvider agentProvider) : AgentExecutor(id: "question_student", session, agentProvider) { // protected override async ValueTask ExecuteAsync(IWorkflowContext context, CancellationToken cancellationToken) { - string? agentName = await context.ReadStateAsync(key: "FOUNDRY_AGENT_RESEARCHMANAGER", scopeName: "Env"); + string? agentName = "StudentAgent"; if (string.IsNullOrWhiteSpace(agentName)) { - throw new InvalidOperationException($"Agent name must be defined: {this.Id}"); + throw new DeclarativeActionException($"Agent name must be defined: {this.Id}"); } - string? conversationId = await context.ReadStateAsync(key: "InternalConversationId", scopeName: "Local"); + string? conversationId = await context.ReadStateAsync(key: "ConversationId", scopeName: "System").ConfigureAwait(false); bool autoSend = true; - string additionalInstructions = - await context.FormatTemplateAsync( - """ - Your only job is to devise an efficient plan that identifies (by name) how a team member may contribute to addressing the user request. - - Only select the following team which is listed as "- [Name]: [Description]" - - {Local.TeamDescription} - - The plan must be a bullet point list must be in the form "- [AgentName]: [Specific action or task for that agent to perform]" - - Remember, there is no requirement to involve the entire team -- only select team member's whose particular expertise is required for this task. - """); - IList? inputMessages = await context.EvaluateListAsync("UserMessage(Local.InputTask)"); - - AgentRunResponse agentResponse = - await InvokeAgentAsync( - context, - agentName, - conversationId, - autoSend, - additionalInstructions, - inputMessages, - cancellationToken); - - if (autoSend) - { - await context.AddEventAsync(new AgentRunResponseEvent(this.Id, agentResponse)); - } - - await context.QueueStateUpdateAsync(key: "Plan", value: agentResponse.Messages, scopeName: "Local"); - - return default; - } - } - - /// - /// Assigns an evaluated expression, other variable, or literal value to the "Local.TaskInstructions" variable. - /// - internal sealed class SetvariableKk2ldlExecutor(FormulaSession session) : ActionExecutor(id: "setVariable_Kk2LDL", session) - { - // - protected override async ValueTask ExecuteAsync(IWorkflowContext context, CancellationToken cancellationToken) - { - object? evaluatedValue = await context.EvaluateValueAsync(""" - "# TASK - Address the following user request: - - " & Local.InputTask & " - - - # TEAM - Use the following team to answer this request: - - " & Local.TeamDescription & " - - - # FACTS - Consider this initial fact sheet: - - " & Trim(Last(Local.TaskFacts).Text) & " - - - # PLAN - Here is the plan to follow as best as possible: - - " & Last(Local.Plan).Text - """); - await context.QueueStateUpdateAsync(key: "TaskInstructions", value: evaluatedValue, scopeName: "Local"); - - return default; - } - } - - /// - /// Formats a message template and sends an activity event. - /// - internal sealed class SendactivityBwnzimExecutor(FormulaSession session) : ActionExecutor(id: "sendActivity_bwNZiM", session) - { - // - protected override async ValueTask ExecuteAsync(IWorkflowContext context, CancellationToken cancellationToken) - { - string activityText = - await context.FormatTemplateAsync( - """ - {Local.TaskInstructions} - """ - ); - AgentRunResponse response = new([new ChatMessage(ChatRole.Assistant, activityText)]); - await context.AddEventAsync(new AgentRunResponseEvent(this.Id, response)); - - return default; - } - } - - /// - /// Invokes an agent to process messages and return a response within a conversation context. - /// - internal sealed class QuestionO3bqkfExecutor(FormulaSession session, WorkflowAgentProvider agentProvider) : AgentExecutor(id: "question_o3BQkf", session, agentProvider) - { - // - protected override async ValueTask ExecuteAsync(IWorkflowContext context, CancellationToken cancellationToken) - { - string? agentName = await context.ReadStateAsync(key: "FOUNDRY_AGENT_RESEARCHMANAGER", scopeName: "Env"); - - if (string.IsNullOrWhiteSpace(agentName)) - { - throw new InvalidOperationException($"Agent name must be defined: {this.Id}"); - } - - string? conversationId = await context.ReadStateAsync(key: "InternalConversationId", scopeName: "Local"); - bool autoSend = true; - string additionalInstructions = - await context.FormatTemplateAsync( - """ - Recall we are working on the following request: - - {Local.InputTask} - - And we have assembled the following team: - - {Local.TeamDescription} - - To make progress on the request, please answer the following questions, including necessary reasoning: - - - Is the request fully satisfied? (True if complete, or False if the original request has yet to be SUCCESSFULLY and FULLY addressed) - - Are we in a loop where we are repeating the same requests and / or getting the same responses from an agent multiple times? Loops can span multiple turns, and can include repeated actions like scrolling up or down more than a handful of times. - - Are we making forward progress? (True if just starting, or recent messages are adding value. False if recent messages show evidence of being stuck in a loop or if there is evidence of significant barriers to success such as the inability to read from a required file) - - Who should speak next? (select from: {Concat(Local.AvailableAgents, name, ",")}) - - What instruction or question would you give this team member? (Phrase as if speaking directly to them, and include any specific information they may need) - - Please output an answer in pure JSON format according to the following schema. The JSON object must be parsable as-is. DO NOT OUTPUT ANYTHING OTHER THAN JSON, AND DO NOT DEVIATE FROM THIS SCHEMA: - - {{ - "is_request_satisfied": {{ - "reason": string, - "answer": boolean - }}, - "is_in_loop": {{ - "reason": string, - "answer": boolean - }}, - "is_progress_being_made": {{ - "reason": string, - "answer": boolean - }}, - "next_speaker": {{ - "reason": string, - "answer": string (select from: {Concat(Local.AvailableAgents, name, ",")}) - }}, - "instruction_or_question": {{ - "reason": string, - "answer": string - }} - }} - """); - IList? inputMessages = await context.EvaluateListAsync("UserMessage(Local.AgentResponseText)"); + IList? inputMessages = null; AgentRunResponse agentResponse = await InvokeAgentAsync( @@ -451,99 +71,14 @@ await InvokeAgentAsync( agentName, conversationId, autoSend, - additionalInstructions, inputMessages, - cancellationToken); + cancellationToken).ConfigureAwait(false); if (autoSend) { - await context.AddEventAsync(new AgentRunResponseEvent(this.Id, agentResponse)); - } - - await context.QueueStateUpdateAsync(key: "ProgressLedgerUpdate", value: agentResponse.Messages, scopeName: "Local"); - - return default; - } - } - - /// - /// Parses a string or untyped value to the provided data type. When the input is a string, it will be treated as JSON. - /// - internal sealed class ParseRnztlvExecutor(FormulaSession session) : ActionExecutor(id: "parse_rNZtlV", session) - { - // - protected override async ValueTask ExecuteAsync(IWorkflowContext context, CancellationToken cancellationToken) - { - VariableType targetType = - VariableType.Record( - ("is_progress_being_made", - VariableType.Record( - ("reason", typeof(string)), - ("answer", typeof(bool)))), - ("is_request_satisfied", - VariableType.Record( - ("reason", typeof(string)), - ("answer", typeof(bool)))), - ("is_in_loop", - VariableType.Record( - ("reason", typeof(string)), - ("answer", typeof(bool)))), - ("next_speaker", - VariableType.Record( - ("reason", typeof(string)), - ("answer", typeof(string)))), - ("instruction_or_question", - VariableType.Record( - ("reason", typeof(string)), - ("answer", typeof(string))))); - object? parsedValue = await context.ConvertValueAsync(targetType, "Last(Local.ProgressLedgerUpdate).Text", cancellationToken); - await context.QueueStateUpdateAsync(key: "TypedProgressLedger", value: parsedValue, scopeName: "Local"); - - return default; - } - } - - /// - /// Conditional branching similar to an if / elseif / elseif / else chain. - /// - internal sealed class ConditiongroupMvieccExecutor(FormulaSession session) : ActionExecutor(id: "conditionGroup_mVIecC", session) - { - // - protected override async ValueTask ExecuteAsync(IWorkflowContext context, CancellationToken cancellationToken) - { - bool condition0 = await context.EvaluateValueAsync("Local.TypedProgressLedger.is_request_satisfied.answer"); - if (condition0) - { - return "conditionItem_fj432c"; - } - - bool condition1 = await context.EvaluateValueAsync("Local.TypedProgressLedger.is_in_loop.answer || Not(Local.TypedProgressLedger.is_progress_being_made.answer)"); - if (condition1) - { - return "conditionItem_yiqund"; + await context.AddEventAsync(new AgentRunResponseEvent(this.Id, agentResponse)).ConfigureAwait(false); } - return "conditionGroup_mVIecCElseActions"; - } - } - - /// - /// Formats a message template and sends an activity event. - /// - internal sealed class SendactivityKdl3mcExecutor(FormulaSession session) : ActionExecutor(id: "sendActivity_kdl3mC", session) - { - // - protected override async ValueTask ExecuteAsync(IWorkflowContext context, CancellationToken cancellationToken) - { - string activityText = - await context.FormatTemplateAsync( - """ - Completed! {Local.TypedProgressLedger.is_request_satisfied.reason} - """ - ); - AgentRunResponse response = new([new ChatMessage(ChatRole.Assistant, activityText)]); - await context.AddEventAsync(new AgentRunResponseEvent(this.Id, response)); - return default; } } @@ -551,28 +86,21 @@ await context.FormatTemplateAsync( /// /// Invokes an agent to process messages and return a response within a conversation context. /// - internal sealed class QuestionKe3l1dExecutor(FormulaSession session, WorkflowAgentProvider agentProvider) : AgentExecutor(id: "question_Ke3l1d", session, agentProvider) + internal sealed class QuestionTeacherExecutor(FormulaSession session, WorkflowAgentProvider agentProvider) : AgentExecutor(id: "question_teacher", session, agentProvider) { // protected override async ValueTask ExecuteAsync(IWorkflowContext context, CancellationToken cancellationToken) { - string? agentName = await context.ReadStateAsync(key: "FOUNDRY_AGENT_RESEARCHMANAGER", scopeName: "Env"); + string? agentName = "TeacherAgent"; if (string.IsNullOrWhiteSpace(agentName)) { - throw new InvalidOperationException($"Agent name must be defined: {this.Id}"); + throw new DeclarativeActionException($"Agent name must be defined: {this.Id}"); } - string? conversationId = await context.ReadStateAsync(key: "ConversationId", scopeName: "System"); - bool autoSend = true; - string additionalInstructions = - await context.FormatTemplateAsync( - """ - We have completed the task. - Based only on the conversation and without adding any new information, synthesize the result of the conversation as a complete response to the user task. - The user will only every see this last response and not the entire conversation, so please ensure it is complete and self-contained. - """); - IList? inputMessages = await context.ReadListAsync(key: "SeedTask", scopeName: "Local"); + string? conversationId = await context.ReadStateAsync(key: "ConversationId", scopeName: "System").ConfigureAwait(false); + bool autoSend = false; + IList? inputMessages = null; AgentRunResponse agentResponse = await InvokeAgentAsync( @@ -580,31 +108,30 @@ await InvokeAgentAsync( agentName, conversationId, autoSend, - additionalInstructions, inputMessages, - cancellationToken); + cancellationToken).ConfigureAwait(false); if (autoSend) { - await context.AddEventAsync(new AgentRunResponseEvent(this.Id, agentResponse)); + await context.AddEventAsync(new AgentRunResponseEvent(this.Id, agentResponse)).ConfigureAwait(false); } - await context.QueueStateUpdateAsync(key: "FinalResponse", value: agentResponse.Messages, scopeName: "Local"); + await context.QueueStateUpdateAsync(key: "TeacherResponse", value: agentResponse.Messages, scopeName: "Local").ConfigureAwait(false); return default; } } /// - /// Assigns an evaluated expression, other variable, or literal value to the "Local.StallCount" variable. + /// Assigns an evaluated expression, other variable, or literal value to the "Local.TurnCount" variable. /// - internal sealed class SetvariableH5lxddExecutor(FormulaSession session) : ActionExecutor(id: "setVariable_H5lXdD", session) + internal sealed class SetCountIncrementExecutor(FormulaSession session) : ActionExecutor(id: "set_count_increment", session) { // protected override async ValueTask ExecuteAsync(IWorkflowContext context, CancellationToken cancellationToken) { - object? evaluatedValue = await context.EvaluateValueAsync("Local.StallCount + 1"); - await context.QueueStateUpdateAsync(key: "StallCount", value: evaluatedValue, scopeName: "Local"); + object? evaluatedValue = await context.EvaluateValueAsync("Local.TurnCount + 1").ConfigureAwait(false); + await context.QueueStateUpdateAsync(key: "TurnCount", value: evaluatedValue, scopeName: "Local").ConfigureAwait(false); return default; } @@ -613,31 +140,31 @@ internal sealed class SetvariableH5lxddExecutor(FormulaSession session) : Action /// /// Conditional branching similar to an if / elseif / elseif / else chain. /// - internal sealed class ConditiongroupVbtqd3Executor(FormulaSession session) : ActionExecutor(id: "conditionGroup_vBTQd3", session) + internal sealed class CheckCompletionExecutor(FormulaSession session) : ActionExecutor(id: "check_completion", session) { // protected override async ValueTask ExecuteAsync(IWorkflowContext context, CancellationToken cancellationToken) { - bool condition0 = await context.EvaluateValueAsync(".TypedProgressLedger.is_in_loop.answer"); + bool condition0 = await context.EvaluateValueAsync("""!IsBlank(Find("CONGRATULATIONS", Upper(Last(Local.TeacherResponse).Text)))""").ConfigureAwait(false); if (condition0) { - return "conditionItem_fpaNL9"; + return "check_turn_done"; } - bool condition1 = await context.EvaluateValueAsync("Not(Local.TypedProgressLedger.is_progress_being_made.answer)"); + bool condition1 = await context.EvaluateValueAsync("Local.TurnCount < 4").ConfigureAwait(false); if (condition1) { - return "conditionItem_NnqvXh"; + return "check_turn_count"; } - return "conditionGroup_vBTQd3ElseActions"; + return "check_completionElseActions"; } } /// /// Formats a message template and sends an activity event. /// - internal sealed class SendactivityFpanl9Executor(FormulaSession session) : ActionExecutor(id: "sendActivity_fpaNL9", session) + internal sealed class SendactivityDoneExecutor(FormulaSession session) : ActionExecutor(id: "sendActivity_done", session) { // protected override async ValueTask ExecuteAsync(IWorkflowContext context, CancellationToken cancellationToken) @@ -645,11 +172,11 @@ internal sealed class SendactivityFpanl9Executor(FormulaSession session) : Actio string activityText = await context.FormatTemplateAsync( """ - {Local.TypedProgressLedger.is_in_loop.reason} + GOLD STAR! """ ); AgentRunResponse response = new([new ChatMessage(ChatRole.Assistant, activityText)]); - await context.AddEventAsync(new AgentRunResponseEvent(this.Id, response)); + await context.AddEventAsync(new AgentRunResponseEvent(this.Id, response)).ConfigureAwait(false); return default; } @@ -658,7 +185,7 @@ await context.FormatTemplateAsync( /// /// Formats a message template and sends an activity event. /// - internal sealed class SendactivityNnqvxhExecutor(FormulaSession session) : ActionExecutor(id: "sendActivity_NnqvXh", session) + internal sealed class SendactivityTiredExecutor(FormulaSession session) : ActionExecutor(id: "sendActivity_tired", session) { // protected override async ValueTask ExecuteAsync(IWorkflowContext context, CancellationToken cancellationToken) @@ -666,488 +193,11 @@ internal sealed class SendactivityNnqvxhExecutor(FormulaSession session) : Actio string activityText = await context.FormatTemplateAsync( """ - {Local.TypedProgressLedger.is_progress_being_made.reason} + Let's try again later... """ ); AgentRunResponse response = new([new ChatMessage(ChatRole.Assistant, activityText)]); - await context.AddEventAsync(new AgentRunResponseEvent(this.Id, response)); - - return default; - } - } - - /// - /// Conditional branching similar to an if / elseif / elseif / else chain. - /// - internal sealed class ConditiongroupXznrdmExecutor(FormulaSession session) : ActionExecutor(id: "conditionGroup_xzNrdM", session) - { - // - protected override async ValueTask ExecuteAsync(IWorkflowContext context, CancellationToken cancellationToken) - { - bool condition0 = await context.EvaluateValueAsync("Local.StallCount > 2"); - if (condition0) - { - return "conditionItem_NlQTBv"; - } - - return "conditionGroup_xzNrdMElseActions"; - } - } - - /// - /// Formats a message template and sends an activity event. - /// - internal sealed class SendactivityH5lxddExecutor(FormulaSession session) : ActionExecutor(id: "sendActivity_H5lXdD", session) - { - // - protected override async ValueTask ExecuteAsync(IWorkflowContext context, CancellationToken cancellationToken) - { - string activityText = - await context.FormatTemplateAsync( - """ - Unable to make sufficient progress... - """ - ); - AgentRunResponse response = new([new ChatMessage(ChatRole.Assistant, activityText)]); - await context.AddEventAsync(new AgentRunResponseEvent(this.Id, response)); - - return default; - } - } - - /// - /// Conditional branching similar to an if / elseif / elseif / else chain. - /// - internal sealed class Conditiongroup4S1z27Executor(FormulaSession session) : ActionExecutor(id: "conditionGroup_4s1Z27", session) - { - // - protected override async ValueTask ExecuteAsync(IWorkflowContext context, CancellationToken cancellationToken) - { - bool condition0 = await context.EvaluateValueAsync("Local.RestartCount > 2"); - if (condition0) - { - return "conditionItem_EXAlhZ"; - } - - return "conditionGroup_4s1Z27ElseActions"; - } - } - - /// - /// Formats a message template and sends an activity event. - /// - internal sealed class SendactivityXkxfuuExecutor(FormulaSession session) : ActionExecutor(id: "sendActivity_xKxFUU", session) - { - // - protected override async ValueTask ExecuteAsync(IWorkflowContext context, CancellationToken cancellationToken) - { - string activityText = - await context.FormatTemplateAsync( - """ - Stopping after attempting {Local.RestartCount} restarts... - """ - ); - AgentRunResponse response = new([new ChatMessage(ChatRole.Assistant, activityText)]); - await context.AddEventAsync(new AgentRunResponseEvent(this.Id, response)); - - return default; - } - } - - /// - /// Formats a message template and sends an activity event. - /// - internal sealed class SendactivityCwnzimExecutor(FormulaSession session) : ActionExecutor(id: "sendActivity_cwNZiM", session) - { - // - protected override async ValueTask ExecuteAsync(IWorkflowContext context, CancellationToken cancellationToken) - { - string activityText = - await context.FormatTemplateAsync( - """ - Re-analyzing facts... - """ - ); - AgentRunResponse response = new([new ChatMessage(ChatRole.Assistant, activityText)]); - await context.AddEventAsync(new AgentRunResponseEvent(this.Id, response)); - - return default; - } - } - - /// - /// Invokes an agent to process messages and return a response within a conversation context. - /// - internal sealed class QuestionWfj123Executor(FormulaSession session, WorkflowAgentProvider agentProvider) : AgentExecutor(id: "question_wFJ123", session, agentProvider) - { - // - protected override async ValueTask ExecuteAsync(IWorkflowContext context, CancellationToken cancellationToken) - { - string? agentName = await context.ReadStateAsync(key: "FOUNDRY_AGENT_RESEARCHANALYST", scopeName: "Env"); - - if (string.IsNullOrWhiteSpace(agentName)) - { - throw new InvalidOperationException($"Agent name must be defined: {this.Id}"); - } - - string? conversationId = await context.ReadStateAsync(key: "InternalConversationId", scopeName: "Local"); - bool autoSend = true; - string additionalInstructions = - await context.FormatTemplateAsync( - """ - It's clear we aren't making as much progress as we would like, but we may have learned something new. - Please rewrite the following fact sheet, updating it to include anything new we have learned that may be helpful. - Example edits can include (but are not limited to) adding new guesses, moving educated guesses to verified facts if appropriate, etc. - Updates may be made to any section of the fact sheet, and more than one section of the fact sheet can be edited. - This is an especially good time to update educated guesses, so please at least add or update one educated guess or hunch, and explain your reasoning. - - Here is the old fact sheet: - - {Local.TaskFacts} - """); - IList? inputMessages = await context.EvaluateListAsync(""" - UserMessage( - "As a reminder, we are working to solve the following task: - - " & Local.InputTask) - """); - - AgentRunResponse agentResponse = - await InvokeAgentAsync( - context, - agentName, - conversationId, - autoSend, - additionalInstructions, - inputMessages, - cancellationToken); - - if (autoSend) - { - await context.AddEventAsync(new AgentRunResponseEvent(this.Id, agentResponse)); - } - - await context.QueueStateUpdateAsync(key: "TaskFacts", value: agentResponse.Messages, scopeName: "Local"); - - return default; - } - } - - /// - /// Formats a message template and sends an activity event. - /// - internal sealed class SendactivityDsbajuExecutor(FormulaSession session) : ActionExecutor(id: "sendActivity_dsBaJU", session) - { - // - protected override async ValueTask ExecuteAsync(IWorkflowContext context, CancellationToken cancellationToken) - { - string activityText = - await context.FormatTemplateAsync( - """ - Re-analyzing plan... - """ - ); - AgentRunResponse response = new([new ChatMessage(ChatRole.Assistant, activityText)]); - await context.AddEventAsync(new AgentRunResponseEvent(this.Id, response)); - - return default; - } - } - - /// - /// Invokes an agent to process messages and return a response within a conversation context. - /// - internal sealed class QuestionUej456Executor(FormulaSession session, WorkflowAgentProvider agentProvider) : AgentExecutor(id: "question_uEJ456", session, agentProvider) - { - // - protected override async ValueTask ExecuteAsync(IWorkflowContext context, CancellationToken cancellationToken) - { - string? agentName = await context.ReadStateAsync(key: "FOUNDRY_AGENT_RESEARCHMANAGER", scopeName: "Env"); - - if (string.IsNullOrWhiteSpace(agentName)) - { - throw new InvalidOperationException($"Agent name must be defined: {this.Id}"); - } - - string? conversationId = await context.ReadStateAsync(key: "InternalConversationId", scopeName: "Local"); - bool autoSend = true; - string additionalInstructions = - await context.FormatTemplateAsync( - """ - Please briefly explain what went wrong on this last run (the root cause of the failure), - and then come up with a new plan that takes steps and/or includes hints to overcome prior challenges and especially avoids repeating the same mistakes. - As before, the new plan should be concise, be expressed in bullet-point form, and consider the following team composition - (do not involve any other outside people since we cannot contact anyone else): - - {Local.TeamDescription} - """); - IList? inputMessages = null; - - AgentRunResponse agentResponse = - await InvokeAgentAsync( - context, - agentName, - conversationId, - autoSend, - additionalInstructions, - inputMessages, - cancellationToken); - - if (autoSend) - { - await context.AddEventAsync(new AgentRunResponseEvent(this.Id, agentResponse)); - } - - await context.QueueStateUpdateAsync(key: "Plan", value: agentResponse.Messages, scopeName: "Local"); - - return default; - } - } - - /// - /// Assigns an evaluated expression, other variable, or literal value to the "Local.TaskInstructions" variable. - /// - internal sealed class SetvariableJw7tmmExecutor(FormulaSession session) : ActionExecutor(id: "setVariable_jW7tmM", session) - { - // - protected override async ValueTask ExecuteAsync(IWorkflowContext context, CancellationToken cancellationToken) - { - object? evaluatedValue = await context.EvaluateValueAsync(""" - "# TASK - Address the following user request: - - " & Local.InputTask & " - - - # TEAM - Use the following team to answer this request: - - " & Local.TeamDescription & " - - - # FACTS - Consider this initial fact sheet: - - " & Local.TaskFacts.Text & " - - - # PLAN - Here is the plan to follow as best as possible: - - " & Local.Plan.Text - """); - await context.QueueStateUpdateAsync(key: "TaskInstructions", value: evaluatedValue, scopeName: "Local"); - - return default; - } - } - - /// - /// Assigns an evaluated expression, other variable, or literal value to the "Local.StallCount" variable. - /// - internal sealed class Setvariable6J2snpExecutor(FormulaSession session) : ActionExecutor(id: "setVariable_6J2snP", session) - { - // - protected override async ValueTask ExecuteAsync(IWorkflowContext context, CancellationToken cancellationToken) - { - object? evaluatedValue = 0; - await context.QueueStateUpdateAsync(key: "StallCount", value: evaluatedValue, scopeName: "Local"); - - return default; - } - } - - /// - /// Assigns an evaluated expression, other variable, or literal value to the "Local.RestartCount" variable. - /// - internal sealed class SetvariableS6hcghExecutor(FormulaSession session) : ActionExecutor(id: "setVariable_S6HCgh", session) - { - // - protected override async ValueTask ExecuteAsync(IWorkflowContext context, CancellationToken cancellationToken) - { - object? evaluatedValue = await context.EvaluateValueAsync("Local.RestartCount + 1"); - await context.QueueStateUpdateAsync(key: "RestartCount", value: evaluatedValue, scopeName: "Local"); - - return default; - } - } - - /// - /// Formats a message template and sends an activity event. - /// - internal sealed class SendactivityL7ooqoExecutor(FormulaSession session) : ActionExecutor(id: "sendActivity_L7ooQO", session) - { - // - protected override async ValueTask ExecuteAsync(IWorkflowContext context, CancellationToken cancellationToken) - { - string activityText = - await context.FormatTemplateAsync( - """ - ({Local.TypedProgressLedger.next_speaker.reason}) - - {Local.TypedProgressLedger.next_speaker.answer} - {Local.TypedProgressLedger.instruction_or_question.answer} - """ - ); - AgentRunResponse response = new([new ChatMessage(ChatRole.Assistant, activityText)]); - await context.AddEventAsync(new AgentRunResponseEvent(this.Id, response)); - - return default; - } - } - - /// - /// Assigns an evaluated expression, other variable, or literal value to the "Local.StallCount" variable. - /// - internal sealed class SetvariableL7ooqoExecutor(FormulaSession session) : ActionExecutor(id: "setVariable_L7ooQO", session) - { - // - protected override async ValueTask ExecuteAsync(IWorkflowContext context, CancellationToken cancellationToken) - { - object? evaluatedValue = 0; - await context.QueueStateUpdateAsync(key: "StallCount", value: evaluatedValue, scopeName: "Local"); - - return default; - } - } - - /// - /// Assigns an evaluated expression, other variable, or literal value to the "Local.NextSpeaker" variable. - /// - internal sealed class SetvariableNxn1meExecutor(FormulaSession session) : ActionExecutor(id: "setVariable_nxN1mE", session) - { - // - protected override async ValueTask ExecuteAsync(IWorkflowContext context, CancellationToken cancellationToken) - { - object? evaluatedValue = await context.EvaluateValueAsync("Search(Local.AvailableAgents, Local.TypedProgressLedger.next_speaker.answer, name)"); - await context.QueueStateUpdateAsync(key: "NextSpeaker", value: evaluatedValue, scopeName: "Local"); - - return default; - } - } - - /// - /// Conditional branching similar to an if / elseif / elseif / else chain. - /// - internal sealed class ConditiongroupQfpif5Executor(FormulaSession session) : ActionExecutor(id: "conditionGroup_QFPiF5", session) - { - // - protected override async ValueTask ExecuteAsync(IWorkflowContext context, CancellationToken cancellationToken) - { - bool condition0 = await context.EvaluateValueAsync("CountRows(Local.NextSpeaker) = 1"); - if (condition0) - { - return "conditionItem_GmigcU"; - } - - return "conditionGroup_QFPiF5ElseActions"; - } - } - - /// - /// Invokes an agent to process messages and return a response within a conversation context. - /// - internal sealed class QuestionOrsbf06Executor(FormulaSession session, WorkflowAgentProvider agentProvider) : AgentExecutor(id: "question_orsBf06", session, agentProvider) - { - // - protected override async ValueTask ExecuteAsync(IWorkflowContext context, CancellationToken cancellationToken) - { - string? agentName = await context.EvaluateValueAsync("First(Local.NextSpeaker).agentid"); - - if (string.IsNullOrWhiteSpace(agentName)) - { - throw new InvalidOperationException($"Agent name must be defined: {this.Id}"); - } - - string? conversationId = await context.ReadStateAsync(key: "ConversationId", scopeName: "System"); - bool autoSend = true; - string additionalInstructions = - await context.FormatTemplateAsync( - """ - {Local.TypedProgressLedger.instruction_or_question.answer} - """); - IList? inputMessages = await context.ReadListAsync(key: "SeedTask", scopeName: "Local"); - - AgentRunResponse agentResponse = - await InvokeAgentAsync( - context, - agentName, - conversationId, - autoSend, - additionalInstructions, - inputMessages, - cancellationToken); - - if (autoSend) - { - await context.AddEventAsync(new AgentRunResponseEvent(this.Id, agentResponse)); - } - - await context.QueueStateUpdateAsync(key: "AgentResponse", value: agentResponse.Messages, scopeName: "Local"); - - return default; - } - } - - /// - /// Assigns an evaluated expression, other variable, or literal value to the "Local.AgentResponseText" variable. - /// - internal sealed class SetvariableXznrdmExecutor(FormulaSession session) : ActionExecutor(id: "setVariable_XzNrdM", session) - { - // - protected override async ValueTask ExecuteAsync(IWorkflowContext context, CancellationToken cancellationToken) - { - object? evaluatedValue = await context.EvaluateValueAsync("Last(Local.AgentResponse).Text"); - await context.QueueStateUpdateAsync(key: "AgentResponseText", value: evaluatedValue, scopeName: "Local"); - - return default; - } - } - - /// - /// Resets the value of the "Local.SeedTask" variable, potentially causing re-evaluation - /// of the default value, question or action that provides the value to this variable. - /// - internal sealed class Setvariable8Eix2aExecutor(FormulaSession session) : ActionExecutor(id: "setVariable_8eIx2A", session) - { - protected override async ValueTask ExecuteAsync(IWorkflowContext context, CancellationToken cancellationToken) - { - await context.QueueStateUpdateAsync(key: "SeedTask", value: UnassignedValue.Instance, scopeName: "Local"); - - return default; - } - } - - /// - /// Formats a message template and sends an activity event. - /// - internal sealed class SendactivityBhcsi7Executor(FormulaSession session) : ActionExecutor(id: "sendActivity_BhcsI7", session) - { - // - protected override async ValueTask ExecuteAsync(IWorkflowContext context, CancellationToken cancellationToken) - { - string activityText = - await context.FormatTemplateAsync( - """ - Unable to choose next agent... - """ - ); - AgentRunResponse response = new([new ChatMessage(ChatRole.Assistant, activityText)]); - await context.AddEventAsync(new AgentRunResponseEvent(this.Id, response)); - - return default; - } - } - - /// - /// Assigns an evaluated expression, other variable, or literal value to the "Local.StallCount" variable. - /// - internal sealed class SetvariableBhcsi7Executor(FormulaSession session) : ActionExecutor(id: "setVariable_BhcsI7", session) - { - // - protected override async ValueTask ExecuteAsync(IWorkflowContext context, CancellationToken cancellationToken) - { - object? evaluatedValue = await context.EvaluateValueAsync("Local.StallCount + 1"); - await context.QueueStateUpdateAsync(key: "StallCount", value: evaluatedValue, scopeName: "Local"); + await context.AddEventAsync(new AgentRunResponseEvent(this.Id, response)).ConfigureAwait(false); return default; } @@ -1162,189 +212,56 @@ public static Workflow CreateWorkflow( inputTransform ??= (message) => DeclarativeWorkflowBuilder.DefaultTransform(message); WorkflowDemoRootExecutor workflowDemoRoot = new(options, inputTransform); DelegateExecutor workflowDemo = new(id: "workflow_demo", workflowDemoRoot.Session); - SetvariableAaslmfExecutor setVariableAaslmf = new(workflowDemoRoot.Session); - SetvariableV6yeboExecutor setVariableV6yebo = new(workflowDemoRoot.Session); - SetvariableNz2u0lExecutor setVariableNz2u0l = new(workflowDemoRoot.Session); - Setvariable10U2znExecutor setVariable10U2zn = new(workflowDemoRoot.Session); - SendactivityYfsbryExecutor sendActivityYfsbry = new(workflowDemoRoot.Session); - Conversation1A2b3cExecutor conversation1A2b3c = new(workflowDemoRoot.Session, options.AgentProvider); - QuestionUdomuwExecutor questionUdomuw = new(workflowDemoRoot.Session, options.AgentProvider); - SendactivityYfsbrzExecutor sendActivityYfsbrz = new(workflowDemoRoot.Session); - QuestionDsbajuExecutor questionDsbaju = new(workflowDemoRoot.Session, options.AgentProvider); - SetvariableKk2ldlExecutor setVariableKk2ldl = new(workflowDemoRoot.Session); - SendactivityBwnzimExecutor sendActivityBwnzim = new(workflowDemoRoot.Session); - QuestionO3bqkfExecutor questionO3bqkf = new(workflowDemoRoot.Session, options.AgentProvider); - ParseRnztlvExecutor parseRnztlv = new(workflowDemoRoot.Session); - ConditiongroupMvieccExecutor conditionGroupMviecc = new(workflowDemoRoot.Session); - DelegateExecutor conditionItemFj432c = new(id: "conditionItem_fj432c", workflowDemoRoot.Session); - DelegateExecutor conditionItemYiqund = new(id: "conditionItem_yiqund", workflowDemoRoot.Session); - DelegateExecutor conditionGroupMvieccelseactions = new(id: "conditionGroup_mVIecCElseActions", workflowDemoRoot.Session); - DelegateExecutor conditionItemFj432cactions = new(id: "conditionItem_fj432cActions", workflowDemoRoot.Session); - SendactivityKdl3mcExecutor sendActivityKdl3mc = new(workflowDemoRoot.Session); - QuestionKe3l1dExecutor questionKe3l1d = new(workflowDemoRoot.Session, options.AgentProvider); - DelegateExecutor endSvonsv = new(id: "end_SVoNSV", workflowDemoRoot.Session); - DelegateExecutor conditionItemYiqundactions = new(id: "conditionItem_yiqundActions", workflowDemoRoot.Session); - SetvariableH5lxddExecutor setVariableH5lxdd = new(workflowDemoRoot.Session); - ConditiongroupVbtqd3Executor conditionGroupVbtqd3 = new(workflowDemoRoot.Session); - DelegateExecutor conditionItemFpanl9 = new(id: "conditionItem_fpaNL9", workflowDemoRoot.Session); - DelegateExecutor conditionItemNnqvxh = new(id: "conditionItem_NnqvXh", workflowDemoRoot.Session); - DelegateExecutor conditionItemFpanl9actions = new(id: "conditionItem_fpaNL9Actions", workflowDemoRoot.Session); - SendactivityFpanl9Executor sendActivityFpanl9 = new(workflowDemoRoot.Session); - DelegateExecutor conditionItemNnqvxhactions = new(id: "conditionItem_NnqvXhActions", workflowDemoRoot.Session); - SendactivityNnqvxhExecutor sendActivityNnqvxh = new(workflowDemoRoot.Session); - DelegateExecutor conditionGroupVbtqd3Post = new(id: "conditionGroup_vBTQd3_Post", workflowDemoRoot.Session); - ConditiongroupXznrdmExecutor conditionGroupXznrdm = new(workflowDemoRoot.Session); - DelegateExecutor conditionItemNlqtbv = new(id: "conditionItem_NlQTBv", workflowDemoRoot.Session); - DelegateExecutor conditionItemNlqtbvactions = new(id: "conditionItem_NlQTBvActions", workflowDemoRoot.Session); - SendactivityH5lxddExecutor sendActivityH5lxdd = new(workflowDemoRoot.Session); - Conditiongroup4S1z27Executor conditionGroup4S1z27 = new(workflowDemoRoot.Session); - DelegateExecutor conditionItemExalhz = new(id: "conditionItem_EXAlhZ", workflowDemoRoot.Session); - DelegateExecutor conditionItemExalhzactions = new(id: "conditionItem_EXAlhZActions", workflowDemoRoot.Session); - SendactivityXkxfuuExecutor sendActivityXkxfuu = new(workflowDemoRoot.Session); - DelegateExecutor endGhvrfh = new(id: "end_GHVrFh", workflowDemoRoot.Session); - DelegateExecutor conditionGroup4S1z27Post = new(id: "conditionGroup_4s1Z27_Post", workflowDemoRoot.Session); - SendactivityCwnzimExecutor sendActivityCwnzim = new(workflowDemoRoot.Session); - QuestionWfj123Executor questionWfj123 = new(workflowDemoRoot.Session, options.AgentProvider); - SendactivityDsbajuExecutor sendActivityDsbaju = new(workflowDemoRoot.Session); - QuestionUej456Executor questionUej456 = new(workflowDemoRoot.Session, options.AgentProvider); - SetvariableJw7tmmExecutor setVariableJw7tmm = new(workflowDemoRoot.Session); - Setvariable6J2snpExecutor setVariable6J2snp = new(workflowDemoRoot.Session); - SetvariableS6hcghExecutor setVariableS6hcgh = new(workflowDemoRoot.Session); - DelegateExecutor gotoLzfj8u = new(id: "goto_LzfJ8u", workflowDemoRoot.Session); - DelegateExecutor conditionItemYiqundRestart = new(id: "conditionItem_yiqund_Restart", workflowDemoRoot.Session); - SendactivityL7ooqoExecutor sendActivityL7ooqo = new(workflowDemoRoot.Session); - SetvariableL7ooqoExecutor setVariableL7ooqo = new(workflowDemoRoot.Session); - DelegateExecutor conditionGroupMvieccPost = new(id: "conditionGroup_mVIecC_Post", workflowDemoRoot.Session); - SetvariableNxn1meExecutor setVariableNxn1me = new(workflowDemoRoot.Session); - ConditiongroupQfpif5Executor conditionGroupQfpif5 = new(workflowDemoRoot.Session); - DelegateExecutor conditionItemGmigcu = new(id: "conditionItem_GmigcU", workflowDemoRoot.Session); - DelegateExecutor conditionGroupQfpif5elseactions = new(id: "conditionGroup_QFPiF5ElseActions", workflowDemoRoot.Session); - DelegateExecutor conditionItemGmigcuactions = new(id: "conditionItem_GmigcUActions", workflowDemoRoot.Session); - QuestionOrsbf06Executor questionOrsbf06 = new(workflowDemoRoot.Session, options.AgentProvider); - SetvariableXznrdmExecutor setVariableXznrdm = new(workflowDemoRoot.Session); - Setvariable8Eix2aExecutor setVariable8Eix2a = new(workflowDemoRoot.Session); - DelegateExecutor conditionItemGmigcuRestart = new(id: "conditionItem_GmigcU_Restart", workflowDemoRoot.Session); - SendactivityBhcsi7Executor sendActivityBhcsi7 = new(workflowDemoRoot.Session); - SetvariableBhcsi7Executor setVariableBhcsi7 = new(workflowDemoRoot.Session); - DelegateExecutor conditionGroupQfpif5Post = new(id: "conditionGroup_QFPiF5_Post", workflowDemoRoot.Session); - DelegateExecutor goto76Hne8 = new(id: "goto_76Hne8", workflowDemoRoot.Session); - DelegateExecutor conditionItemFj432cPost = new(id: "conditionItem_fj432c_Post", workflowDemoRoot.Session); - DelegateExecutor conditionItemYiqundPost = new(id: "conditionItem_yiqund_Post", workflowDemoRoot.Session); - DelegateExecutor endSvonsvRestart = new(id: "end_SVoNSV_Restart", workflowDemoRoot.Session); - DelegateExecutor conditionItemFj432cactionsPost = new(id: "conditionItem_fj432cActions_Post", workflowDemoRoot.Session); - DelegateExecutor conditionGroupXznrdmPost = new(id: "conditionGroup_xzNrdM_Post", workflowDemoRoot.Session); - DelegateExecutor conditionItemYiqundactionsPost = new(id: "conditionItem_yiqundActions_Post", workflowDemoRoot.Session); - DelegateExecutor conditionItemFpanl9Post = new(id: "conditionItem_fpaNL9_Post", workflowDemoRoot.Session); - DelegateExecutor conditionItemNnqvxhPost = new(id: "conditionItem_NnqvXh_Post", workflowDemoRoot.Session); - DelegateExecutor conditionItemFpanl9actionsPost = new(id: "conditionItem_fpaNL9Actions_Post", workflowDemoRoot.Session); - DelegateExecutor conditionItemNnqvxhactionsPost = new(id: "conditionItem_NnqvXhActions_Post", workflowDemoRoot.Session); - DelegateExecutor conditionItemNlqtbvPost = new(id: "conditionItem_NlQTBv_Post", workflowDemoRoot.Session); - DelegateExecutor gotoLzfj8uRestart = new(id: "goto_LzfJ8u_Restart", workflowDemoRoot.Session); - DelegateExecutor conditionItemNlqtbvactionsPost = new(id: "conditionItem_NlQTBvActions_Post", workflowDemoRoot.Session); - DelegateExecutor conditionItemExalhzPost = new(id: "conditionItem_EXAlhZ_Post", workflowDemoRoot.Session); - DelegateExecutor endGhvrfhRestart = new(id: "end_GHVrFh_Restart", workflowDemoRoot.Session); - DelegateExecutor conditionItemExalhzactionsPost = new(id: "conditionItem_EXAlhZActions_Post", workflowDemoRoot.Session); - DelegateExecutor conditionGroupMvieccelseactionsPost = new(id: "conditionGroup_mVIecCElseActions_Post", workflowDemoRoot.Session); - DelegateExecutor conditionItemGmigcuPost = new(id: "conditionItem_GmigcU_Post", workflowDemoRoot.Session); - DelegateExecutor conditionItemGmigcuactionsPost = new(id: "conditionItem_GmigcUActions_Post", workflowDemoRoot.Session); - DelegateExecutor conditionGroupQfpif5elseactionsPost = new(id: "conditionGroup_QFPiF5ElseActions_Post", workflowDemoRoot.Session); + QuestionStudentExecutor questionStudent = new(workflowDemoRoot.Session, options.AgentProvider); + QuestionTeacherExecutor questionTeacher = new(workflowDemoRoot.Session, options.AgentProvider); + SetCountIncrementExecutor setCountIncrement = new(workflowDemoRoot.Session); + CheckCompletionExecutor checkCompletion = new(workflowDemoRoot.Session); + DelegateExecutor checkTurnDone = new(id: "check_turn_done", workflowDemoRoot.Session); + DelegateExecutor checkTurnCount = new(id: "check_turn_count", workflowDemoRoot.Session); + DelegateExecutor checkCompletionelseactions = new(id: "check_completionElseActions", workflowDemoRoot.Session); + DelegateExecutor checkTurnDoneactions = new(id: "check_turn_doneActions", workflowDemoRoot.Session); + SendactivityDoneExecutor sendActivityDone = new(workflowDemoRoot.Session); + DelegateExecutor checkTurnCountactions = new(id: "check_turn_countActions", workflowDemoRoot.Session); + DelegateExecutor gotoStudentAgent = new(id: "goto_student_agent", workflowDemoRoot.Session); + DelegateExecutor checkTurnCountRestart = new(id: "check_turn_count_Restart", workflowDemoRoot.Session); + SendactivityTiredExecutor sendActivityTired = new(workflowDemoRoot.Session); + DelegateExecutor checkTurnDonePost = new(id: "check_turn_done_Post", workflowDemoRoot.Session); + DelegateExecutor checkCompletionPost = new(id: "check_completion_Post", workflowDemoRoot.Session); + DelegateExecutor checkTurnCountPost = new(id: "check_turn_count_Post", workflowDemoRoot.Session); + DelegateExecutor checkTurnDoneactionsPost = new(id: "check_turn_doneActions_Post", workflowDemoRoot.Session); + DelegateExecutor gotoStudentAgentRestart = new(id: "goto_student_agent_Restart", workflowDemoRoot.Session); + DelegateExecutor checkTurnCountactionsPost = new(id: "check_turn_countActions_Post", workflowDemoRoot.Session); + DelegateExecutor checkCompletionelseactionsPost = new(id: "check_completionElseActions_Post", workflowDemoRoot.Session); // Define the workflow builder WorkflowBuilder builder = new(workflowDemoRoot); // Connect executors builder.AddEdge(workflowDemoRoot, workflowDemo); - builder.AddEdge(workflowDemo, setVariableAaslmf); - builder.AddEdge(setVariableAaslmf, setVariableV6yebo); - builder.AddEdge(setVariableV6yebo, setVariableNz2u0l); - builder.AddEdge(setVariableNz2u0l, setVariable10U2zn); - builder.AddEdge(setVariable10U2zn, sendActivityYfsbry); - builder.AddEdge(sendActivityYfsbry, conversation1A2b3c); - builder.AddEdge(conversation1A2b3c, questionUdomuw); - builder.AddEdge(questionUdomuw, sendActivityYfsbrz); - builder.AddEdge(sendActivityYfsbrz, questionDsbaju); - builder.AddEdge(questionDsbaju, setVariableKk2ldl); - builder.AddEdge(setVariableKk2ldl, sendActivityBwnzim); - builder.AddEdge(sendActivityBwnzim, questionO3bqkf); - builder.AddEdge(questionO3bqkf, parseRnztlv); - builder.AddEdge(parseRnztlv, conditionGroupMviecc); - builder.AddEdge(conditionGroupMviecc, conditionItemFj432c, (object? result) => ActionExecutor.IsMatch("conditionItem_fj432c", result)); - builder.AddEdge(conditionGroupMviecc, conditionItemYiqund, (object? result) => ActionExecutor.IsMatch("conditionItem_yiqund", result)); - builder.AddEdge(conditionGroupMviecc, conditionGroupMvieccelseactions, (object? result) => ActionExecutor.IsMatch("conditionGroup_mVIecCElseActions", result)); - builder.AddEdge(conditionItemFj432c, conditionItemFj432cactions); - builder.AddEdge(conditionItemFj432cactions, sendActivityKdl3mc); - builder.AddEdge(sendActivityKdl3mc, questionKe3l1d); - builder.AddEdge(questionKe3l1d, endSvonsv); - builder.AddEdge(conditionItemYiqund, conditionItemYiqundactions); - builder.AddEdge(conditionItemYiqundactions, setVariableH5lxdd); - builder.AddEdge(setVariableH5lxdd, conditionGroupVbtqd3); - builder.AddEdge(conditionGroupVbtqd3, conditionItemFpanl9, (object? result) => ActionExecutor.IsMatch("conditionItem_fpaNL9", result)); - builder.AddEdge(conditionGroupVbtqd3, conditionItemNnqvxh, (object? result) => ActionExecutor.IsMatch("conditionItem_NnqvXh", result)); - builder.AddEdge(conditionItemFpanl9, conditionItemFpanl9actions); - builder.AddEdge(conditionItemFpanl9actions, sendActivityFpanl9); - builder.AddEdge(conditionItemNnqvxh, conditionItemNnqvxhactions); - builder.AddEdge(conditionItemNnqvxhactions, sendActivityNnqvxh); - builder.AddEdge(conditionGroupVbtqd3Post, conditionGroupXznrdm); - builder.AddEdge(conditionGroupXznrdm, conditionItemNlqtbv, (object? result) => ActionExecutor.IsMatch("conditionItem_NlQTBv", result)); - builder.AddEdge(conditionItemNlqtbv, conditionItemNlqtbvactions); - builder.AddEdge(conditionItemNlqtbvactions, sendActivityH5lxdd); - builder.AddEdge(sendActivityH5lxdd, conditionGroup4S1z27); - builder.AddEdge(conditionGroup4S1z27, conditionItemExalhz, (object? result) => ActionExecutor.IsMatch("conditionItem_EXAlhZ", result)); - builder.AddEdge(conditionItemExalhz, conditionItemExalhzactions); - builder.AddEdge(conditionItemExalhzactions, sendActivityXkxfuu); - builder.AddEdge(sendActivityXkxfuu, endGhvrfh); - builder.AddEdge(conditionGroup4S1z27Post, sendActivityCwnzim); - builder.AddEdge(sendActivityCwnzim, questionWfj123); - builder.AddEdge(questionWfj123, sendActivityDsbaju); - builder.AddEdge(sendActivityDsbaju, questionUej456); - builder.AddEdge(questionUej456, setVariableJw7tmm); - builder.AddEdge(setVariableJw7tmm, setVariable6J2snp); - builder.AddEdge(setVariable6J2snp, setVariableS6hcgh); - builder.AddEdge(setVariableS6hcgh, gotoLzfj8u); - builder.AddEdge(gotoLzfj8u, questionO3bqkf); - builder.AddEdge(conditionItemYiqundRestart, conditionGroupMvieccelseactions); - builder.AddEdge(conditionGroupMvieccelseactions, sendActivityL7ooqo); - builder.AddEdge(sendActivityL7ooqo, setVariableL7ooqo); - builder.AddEdge(conditionGroupMvieccPost, setVariableNxn1me); - builder.AddEdge(setVariableNxn1me, conditionGroupQfpif5); - builder.AddEdge(conditionGroupQfpif5, conditionItemGmigcu, (object? result) => ActionExecutor.IsMatch("conditionItem_GmigcU", result)); - builder.AddEdge(conditionGroupQfpif5, conditionGroupQfpif5elseactions, (object? result) => ActionExecutor.IsMatch("conditionGroup_QFPiF5ElseActions", result)); - builder.AddEdge(conditionItemGmigcu, conditionItemGmigcuactions); - builder.AddEdge(conditionItemGmigcuactions, questionOrsbf06); - builder.AddEdge(questionOrsbf06, setVariableXznrdm); - builder.AddEdge(setVariableXznrdm, setVariable8Eix2a); - builder.AddEdge(conditionItemGmigcuRestart, conditionGroupQfpif5elseactions); - builder.AddEdge(conditionGroupQfpif5elseactions, sendActivityBhcsi7); - builder.AddEdge(sendActivityBhcsi7, setVariableBhcsi7); - builder.AddEdge(conditionGroupQfpif5Post, goto76Hne8); - builder.AddEdge(goto76Hne8, questionO3bqkf); - builder.AddEdge(conditionItemFj432cPost, conditionGroupMvieccPost); - builder.AddEdge(conditionItemYiqundPost, conditionGroupMvieccPost); - builder.AddEdge(endSvonsvRestart, conditionItemFj432cactionsPost); - builder.AddEdge(conditionItemFj432cactionsPost, conditionItemFj432cPost); - builder.AddEdge(conditionGroupXznrdmPost, conditionItemYiqundactionsPost); - builder.AddEdge(conditionItemYiqundactionsPost, conditionItemYiqundPost); - builder.AddEdge(conditionItemFpanl9Post, conditionGroupVbtqd3Post); - builder.AddEdge(conditionItemNnqvxhPost, conditionGroupVbtqd3Post); - builder.AddEdge(sendActivityFpanl9, conditionItemFpanl9actionsPost); - builder.AddEdge(conditionItemFpanl9actionsPost, conditionItemFpanl9Post); - builder.AddEdge(sendActivityNnqvxh, conditionItemNnqvxhactionsPost); - builder.AddEdge(conditionItemNnqvxhactionsPost, conditionItemNnqvxhPost); - builder.AddEdge(conditionItemNlqtbvPost, conditionGroupXznrdmPost); - builder.AddEdge(gotoLzfj8uRestart, conditionItemNlqtbvactionsPost); - builder.AddEdge(conditionItemNlqtbvactionsPost, conditionItemNlqtbvPost); - builder.AddEdge(conditionItemExalhzPost, conditionGroup4S1z27Post); - builder.AddEdge(endGhvrfhRestart, conditionItemExalhzactionsPost); - builder.AddEdge(conditionItemExalhzactionsPost, conditionItemExalhzPost); - builder.AddEdge(setVariableL7ooqo, conditionGroupMvieccelseactionsPost); - builder.AddEdge(conditionGroupMvieccelseactionsPost, conditionGroupMvieccPost); - builder.AddEdge(conditionItemGmigcuPost, conditionGroupQfpif5Post); - builder.AddEdge(setVariable8Eix2a, conditionItemGmigcuactionsPost); - builder.AddEdge(conditionItemGmigcuactionsPost, conditionItemGmigcuPost); - builder.AddEdge(setVariableBhcsi7, conditionGroupQfpif5elseactionsPost); - builder.AddEdge(conditionGroupQfpif5elseactionsPost, conditionGroupQfpif5Post); + builder.AddEdge(workflowDemo, questionStudent); + builder.AddEdge(questionStudent, questionTeacher); + builder.AddEdge(questionTeacher, setCountIncrement); + builder.AddEdge(setCountIncrement, checkCompletion); + builder.AddEdge(checkCompletion, checkTurnDone, (object? result) => ActionExecutor.IsMatch("check_turn_done", result)); + builder.AddEdge(checkCompletion, checkTurnCount, (object? result) => ActionExecutor.IsMatch("check_turn_count", result)); + builder.AddEdge(checkCompletion, checkCompletionelseactions, (object? result) => ActionExecutor.IsMatch("check_completionElseActions", result)); + builder.AddEdge(checkTurnDone, checkTurnDoneactions); + builder.AddEdge(checkTurnDoneactions, sendActivityDone); + builder.AddEdge(checkTurnCount, checkTurnCountactions); + builder.AddEdge(checkTurnCountactions, gotoStudentAgent); + builder.AddEdge(gotoStudentAgent, questionStudent); + builder.AddEdge(checkTurnCountRestart, checkCompletionelseactions); + builder.AddEdge(checkCompletionelseactions, sendActivityTired); + builder.AddEdge(checkTurnDonePost, checkCompletionPost); + builder.AddEdge(checkTurnCountPost, checkCompletionPost); + builder.AddEdge(sendActivityDone, checkTurnDoneactionsPost); + builder.AddEdge(checkTurnDoneactionsPost, checkTurnDonePost); + builder.AddEdge(gotoStudentAgentRestart, checkTurnCountactionsPost); + builder.AddEdge(checkTurnCountactionsPost, checkTurnCountPost); + builder.AddEdge(sendActivityTired, checkCompletionelseactionsPost); + builder.AddEdge(checkCompletionelseactionsPost, checkCompletionPost); // Build the workflow - return builder.Build(); + return builder.Build(validateOrphans: false); } } diff --git a/dotnet/samples/GettingStarted/Workflows/Declarative/ExecuteCode/Program.cs b/dotnet/samples/GettingStarted/Workflows/Declarative/ExecuteCode/Program.cs index c1846dac5e..bfc738c336 100644 --- a/dotnet/samples/GettingStarted/Workflows/Declarative/ExecuteCode/Program.cs +++ b/dotnet/samples/GettingStarted/Workflows/Declarative/ExecuteCode/Program.cs @@ -1,14 +1,14 @@ // Copyright (c) Microsoft. All rights reserved. -using System.Diagnostics; +// Uncomment this to enable JSON checkpointing to the local file system. +//#define CHECKPOINT_JSON + using System.Reflection; -using Azure.AI.Agents.Persistent; using Azure.Identity; using Microsoft.Agents.AI.Workflows; using Microsoft.Agents.AI.Workflows.Declarative; -using Microsoft.Extensions.AI; using Microsoft.Extensions.Configuration; -using Test.WorkflowProviders; +using Shared.Workflows; namespace Demo.DeclarativeCode; @@ -24,157 +24,64 @@ internal sealed class Program { public static async Task Main(string[] args) { - Program program = new(args); + string? workflowInput = ParseWorkflowInput(args); + + Program program = new(workflowInput); await program.ExecuteAsync(); } private async Task ExecuteAsync() + { + Notify("\nWORKFLOW: Starting..."); + + string input = this.GetWorkflowInput(); + + // Execute the workflow: The WorkflowRunner demonstrates how to execute + // a workflow, handle the workflow events, and providing external input. + // This also includes the ability to checkpoint workflow state and how to + // resume execution. + await this.Runner.ExecuteAsync(this.CreateWorkflow, input); + + Notify("\nWORKFLOW: Done!\n"); + } + + private Workflow CreateWorkflow() { // Use DeclarativeWorkflowBuilder to build a workflow based on a YAML file. DeclarativeWorkflowOptions options = - new(new AzureAgentProvider(this.FoundryEndpoint, new AzureCliCredential())) + new(new AzureAgentProvider(new Uri(this.FoundryEndpoint), new AzureCliCredential())) { Configuration = this.Configuration }; // Use the generated provider to create a workflow instance. - Workflow workflow = TestWorkflowProvider.CreateWorkflow(options); - - Notify("\nWORKFLOW: Starting..."); - - // Run the workflow, just like any other workflow - string input = this.GetWorkflowInput(); - StreamingRun run = await InProcessExecution.StreamAsync(workflow, input: input); - await this.MonitorAndDisposeWorkflowRunAsync(run); - - Notify("\nWORKFLOW: Done!"); + return SampleWorkflowProvider.CreateWorkflow(options); } - private const string ConfigKeyFoundryEndpoint = "FOUNDRY_PROJECT_ENDPOINT"; - - private static readonly Dictionary s_nameCache = []; - private static readonly HashSet s_fileCache = []; - private string? WorkflowInput { get; } private string FoundryEndpoint { get; } - private PersistentAgentsClient FoundryClient { get; } private IConfiguration Configuration { get; } + private WorkflowRunner Runner { get; } - private Program(string[] args) + private Program(string? workflowInput) { - this.WorkflowInput = ParseWorkflowInput(args); + this.WorkflowInput = workflowInput; this.Configuration = InitializeConfig(); - this.FoundryEndpoint = this.Configuration[ConfigKeyFoundryEndpoint] ?? throw new InvalidOperationException($"Undefined configuration setting: {ConfigKeyFoundryEndpoint}"); - this.FoundryClient = new PersistentAgentsClient(this.FoundryEndpoint, new AzureCliCredential()); - } - - private async Task MonitorAndDisposeWorkflowRunAsync(StreamingRun run) - { - await using IAsyncDisposable disposeRun = run; - - string? messageId = null; + this.FoundryEndpoint = this.Configuration[Application.Settings.FoundryEndpoint] ?? throw new InvalidOperationException($"Undefined configuration setting: {Application.Settings.FoundryEndpoint}"); - await foreach (WorkflowEvent workflowEvent in run.WatchStreamAsync()) - { - switch (workflowEvent) + this.Runner = + new() { - case ExecutorInvokedEvent executorInvoked: - Debug.WriteLine($"STEP ENTER #{executorInvoked.ExecutorId}"); - break; - - case ExecutorCompletedEvent executorComplete: - Debug.WriteLine($"STEP EXIT #{executorComplete.ExecutorId}"); - break; - - case ExecutorFailedEvent executorFailure: - Debug.WriteLine($"STEP ERROR #{executorFailure.ExecutorId}: {executorFailure.Data?.Message ?? "Unknown"}"); - break; - - case WorkflowErrorEvent workflowError: - throw workflowError.Data as Exception ?? new InvalidOperationException("Unexpected failure..."); - - case ConversationUpdateEvent invokeEvent: - Debug.WriteLine($"CONVERSATION: {invokeEvent.Data}"); - break; - - case AgentRunUpdateEvent streamEvent: - if (!string.Equals(messageId, streamEvent.Update.MessageId, StringComparison.Ordinal)) - { - messageId = streamEvent.Update.MessageId; - - if (messageId is not null) - { - string? agentId = streamEvent.Update.AuthorName; - if (agentId is not null) - { - if (!s_nameCache.TryGetValue(agentId, out string? realName)) - { - PersistentAgent agent = await this.FoundryClient.Administration.GetAgentAsync(agentId); - s_nameCache[agentId] = agent.Name; - realName = agent.Name; - } - agentId = realName; - } - agentId ??= nameof(ChatRole.Assistant); - Console.ForegroundColor = ConsoleColor.Cyan; - Console.Write($"\n{agentId.ToUpperInvariant()}:"); - Console.ForegroundColor = ConsoleColor.DarkGray; - Console.WriteLine($" [{messageId}]"); - } - } - - ChatResponseUpdate? chatUpdate = streamEvent.Update.RawRepresentation as ChatResponseUpdate; - switch (chatUpdate?.RawRepresentation) - { - case MessageContentUpdate messageUpdate: - string? fileId = messageUpdate.ImageFileId ?? messageUpdate.TextAnnotation?.OutputFileId; - if (fileId is not null && s_fileCache.Add(fileId)) - { - BinaryData content = await this.FoundryClient.Files.GetFileContentAsync(fileId); - await DownloadFileContentAsync(Path.GetFileName(messageUpdate.TextAnnotation?.TextToReplace ?? "response.png"), content); - } - break; - } - try - { - Console.ResetColor(); - Console.Write(streamEvent.Data); - } - finally - { - Console.ResetColor(); - } - break; - - case AgentRunResponseEvent messageEvent: - try - { - Console.WriteLine(); - if (messageEvent.Response.AgentId is null) - { - Console.ForegroundColor = ConsoleColor.Cyan; - Console.WriteLine("ACTIVITY:"); - Console.ForegroundColor = ConsoleColor.Yellow; - Console.WriteLine(messageEvent.Response?.Text.Trim()); - } - else - { - if (messageEvent.Response.Usage is not null) - { - Console.ForegroundColor = ConsoleColor.DarkGray; - Console.WriteLine($"[Tokens Total: {messageEvent.Response.Usage.TotalTokenCount}, Input: {messageEvent.Response.Usage.InputTokenCount}, Output: {messageEvent.Response.Usage.OutputTokenCount}]"); - } - } - } - finally - { - Console.ResetColor(); - } - break; - } - } +#if CHECKPOINT_JSON + // Use an json file checkpoint store that will persist checkpoints to the local file system. + UseJsonCheckpoints = true +#else + // Use an in-memory checkpoint store that will not persist checkpoints beyond the lifetime of the process. + UseJsonCheckpoints = false +#endif + }; } private string GetWorkflowInput() @@ -231,19 +138,4 @@ private static void Notify(string message) Console.ResetColor(); } } - - private static async ValueTask DownloadFileContentAsync(string filename, BinaryData content) - { - string filePath = Path.Combine(Path.GetTempPath(), Path.GetFileName(filename)); - filePath = Path.ChangeExtension(filePath, ".png"); - - await File.WriteAllBytesAsync(filePath, content.ToArray()); - - Process.Start( - new ProcessStartInfo - { - FileName = "cmd.exe", - Arguments = $"/C start {filePath}" - }); - } } diff --git a/dotnet/samples/GettingStarted/Workflows/Declarative/ExecuteWorkflow/ExecuteWorkflow.csproj b/dotnet/samples/GettingStarted/Workflows/Declarative/ExecuteWorkflow/ExecuteWorkflow.csproj index b885a71c3c..074a31121d 100644 --- a/dotnet/samples/GettingStarted/Workflows/Declarative/ExecuteWorkflow/ExecuteWorkflow.csproj +++ b/dotnet/samples/GettingStarted/Workflows/Declarative/ExecuteWorkflow/ExecuteWorkflow.csproj @@ -2,16 +2,16 @@ Exe - net9.0 - net9.0 - $(ProjectsDebugTargetFrameworks) + net10.0 enable enable $(NoWarn);CA1812 - true + true + true + true @@ -26,6 +26,7 @@ + diff --git a/dotnet/samples/GettingStarted/Workflows/Declarative/ExecuteWorkflow/Program.cs b/dotnet/samples/GettingStarted/Workflows/Declarative/ExecuteWorkflow/Program.cs index ce6a19b0d3..d1f0980ec7 100644 --- a/dotnet/samples/GettingStarted/Workflows/Declarative/ExecuteWorkflow/Program.cs +++ b/dotnet/samples/GettingStarted/Workflows/Declarative/ExecuteWorkflow/Program.cs @@ -5,18 +5,12 @@ using System.Diagnostics; using System.Reflection; -using System.Text.Json; -using Azure.AI.Agents.Persistent; using Azure.Identity; using Microsoft.Agents.AI.Workflows; -#if CHECKPOINT_JSON -using Microsoft.Agents.AI.Workflows.Checkpointing; -#endif using Microsoft.Agents.AI.Workflows.Declarative; -using Microsoft.Agents.AI.Workflows.Declarative.Events; -using Microsoft.Agents.AI.Workflows.Declarative.Kit; using Microsoft.Extensions.AI; using Microsoft.Extensions.Configuration; +using Shared.Workflows; namespace Demo.DeclarativeWorkflow; @@ -62,73 +56,29 @@ private async Task ExecuteAsync() Notify("\nWORKFLOW: Starting..."); - // Run the workflow, just like any other workflow string input = this.GetWorkflowInput(); -#if CHECKPOINT_JSON - // Use a file-system based JSON checkpoint store to persist checkpoints to disk. - DirectoryInfo checkpointFolder = Directory.CreateDirectory(Path.Combine(".", $"chk-{DateTime.Now:yyMMdd-hhmmss-ff}")); - CheckpointManager checkpointManager = CheckpointManager.CreateJson(new FileSystemJsonCheckpointStore(checkpointFolder)); -#else - // Use an in-memory checkpoint store that will not persist checkpoints beyond the lifetime of the process. - CheckpointManager checkpointManager = CheckpointManager.CreateInMemory(); -#endif - - Checkpointed run = await InProcessExecution.StreamAsync(workflow, input, checkpointManager); - - bool isComplete = false; - object? response = null; - do - { - ExternalRequest? externalRequest = await this.MonitorAndDisposeWorkflowRunAsync(run, response); - if (externalRequest is not null) - { - Notify("\nWORKFLOW: Yield"); - - if (this.LastCheckpoint is null) - { - throw new InvalidOperationException("Checkpoint information missing after external request."); - } - - // Process the external request. - response = await this.HandleExternalRequestAsync(externalRequest); - - // Let's resume on an entirely new workflow instance to demonstrate checkpoint portability. - workflow = this.CreateWorkflow(); - - // Restore the latest checkpoint. - Debug.WriteLine($"RESTORE #{this.LastCheckpoint.CheckpointId}"); - Notify("\nWORKFLOW: Restore"); - - run = await InProcessExecution.ResumeStreamAsync(workflow, this.LastCheckpoint, checkpointManager, run.Run.RunId); - } - else - { - isComplete = true; - } - } - while (!isComplete); - - Notify("\nWORKFLOW: Done!\n"); + // Execute the workflow: The WorkflowRunner demonstrates how to execute + // a workflow, handle the workflow events, and providing external input. + // This also includes the ability to checkpoint workflow state and how to + // resume execution. + await this.Runner.ExecuteAsync(this.CreateWorkflow, input); } /// /// Create the workflow from the declarative YAML. Includes definition of the /// and the associated . /// - /// - /// The value assigned to controls on whether the function - /// tools () initialized in the constructor are included for auto-invocation. - /// private Workflow CreateWorkflow() { - // Use DeclarativeWorkflowBuilder to build a workflow based on a YAML file. - AzureAgentProvider agentProvider = new(this.FoundryEndpoint, new AzureCliCredential()) + // Create the agent provider that will service agent requests within the workflow. + AzureAgentProvider agentProvider = new(new Uri(this.FoundryEndpoint), new AzureCliCredential()) { // Functions included here will be auto-executed by the framework. - Functions = IncludeFunctions ? this.FunctionMap.Values : null, + Functions = this.Functions }; + // Define the workflow options. DeclarativeWorkflowOptions options = new(agentProvider) { @@ -137,31 +87,16 @@ private Workflow CreateWorkflow() //LoggerFactory = null, // Assign to enable logging }; + // Use DeclarativeWorkflowBuilder to build a workflow based on a YAML file. return DeclarativeWorkflowBuilder.Build(this.WorkflowFile, options); } - /// - /// Configuration key used to identify the Foundry project endpoint. - /// - private const string ConfigKeyFoundryEndpoint = "FOUNDRY_PROJECT_ENDPOINT"; - - /// - /// Controls on whether the function tools () initialized - /// in the constructor are included for auto-invocation. - /// NOTE: By default, no functions exist as part of this sample. - /// - private const bool IncludeFunctions = true; - - private static Dictionary NameCache { get; } = []; - private static HashSet FileCache { get; } = []; - private string WorkflowFile { get; } private string? WorkflowInput { get; } private string FoundryEndpoint { get; } - private PersistentAgentsClient FoundryClient { get; } private IConfiguration Configuration { get; } - private CheckpointInfo? LastCheckpoint { get; set; } - private Dictionary FunctionMap { get; } + private WorkflowRunner Runner { get; } + private IList Functions { get; } private Program(string workflowFile, string? workflowInput) { @@ -170,246 +105,26 @@ private Program(string workflowFile, string? workflowInput) this.Configuration = InitializeConfig(); - this.FoundryEndpoint = this.Configuration[ConfigKeyFoundryEndpoint] ?? throw new InvalidOperationException($"Undefined configuration setting: {ConfigKeyFoundryEndpoint}"); - this.FoundryClient = new PersistentAgentsClient(this.FoundryEndpoint, new AzureCliCredential()); + this.FoundryEndpoint = this.Configuration[Application.Settings.FoundryEndpoint] ?? throw new InvalidOperationException($"Undefined configuration setting: {Application.Settings.FoundryEndpoint}"); - List functions = + this.Functions = [ // Manually define any custom functions that may be required by agents within the workflow. // By default, this sample does not include any functions. //AIFunctionFactory.Create(), ]; - this.FunctionMap = functions.ToDictionary(f => f.Name); - } - private async Task MonitorAndDisposeWorkflowRunAsync(Checkpointed run, object? response = null) - { - // Always dispose the run when done. - await using IAsyncDisposable disposeRun = run; - - bool hasStreamed = false; - string? messageId = null; - - await foreach (WorkflowEvent workflowEvent in run.Run.WatchStreamAsync()) - { - switch (workflowEvent) + this.Runner = + new(this.Functions) { - case ExecutorInvokedEvent executorInvoked: - Debug.WriteLine($"EXECUTOR ENTER #{executorInvoked.ExecutorId}"); - break; - - case ExecutorCompletedEvent executorCompleted: - Debug.WriteLine($"EXECUTOR EXIT #{executorCompleted.ExecutorId}"); - break; - - case DeclarativeActionInvokedEvent actionInvoked: - Debug.WriteLine($"ACTION ENTER #{actionInvoked.ActionId} [{actionInvoked.ActionType}]"); - break; - - case DeclarativeActionCompletedEvent actionComplete: - Debug.WriteLine($"ACTION EXIT #{actionComplete.ActionId} [{actionComplete.ActionType}]"); - break; - - case ExecutorFailedEvent executorFailure: - Debug.WriteLine($"STEP ERROR #{executorFailure.ExecutorId}: {executorFailure.Data?.Message ?? "Unknown"}"); - break; - - case WorkflowErrorEvent workflowError: - throw workflowError.Data as Exception ?? new InvalidOperationException("Unexpected failure..."); - - case SuperStepCompletedEvent checkpointCompleted: - this.LastCheckpoint = checkpointCompleted.CompletionInfo?.Checkpoint; - Debug.WriteLine($"CHECKPOINT x{checkpointCompleted.StepNumber} [{this.LastCheckpoint?.CheckpointId ?? "(none)"}]"); - break; - - case RequestInfoEvent requestInfo: - Debug.WriteLine($"REQUEST #{requestInfo.Request.RequestId}"); - if (response is not null) - { - ExternalResponse requestResponse = requestInfo.Request.CreateResponse(response); - await run.Run.SendResponseAsync(requestResponse); - response = null; - } - else - { - // Yield to handle the external request - return requestInfo.Request; - } - break; - - case ConversationUpdateEvent invokeEvent: - Debug.WriteLine($"CONVERSATION: {invokeEvent.Data}"); - break; - - case MessageActivityEvent activityEvent: - Console.ForegroundColor = ConsoleColor.Cyan; - Console.WriteLine("\nACTIVITY:"); - Console.ForegroundColor = ConsoleColor.Yellow; - Console.WriteLine(activityEvent.Message.Trim()); - break; - - case AgentRunUpdateEvent streamEvent: - if (!string.Equals(messageId, streamEvent.Update.MessageId, StringComparison.Ordinal)) - { - hasStreamed = false; - messageId = streamEvent.Update.MessageId; - - if (messageId is not null) - { - string? agentId = streamEvent.Update.AgentId; - if (agentId is not null) - { - if (!NameCache.TryGetValue(agentId, out string? realName)) - { - PersistentAgent agent = await this.FoundryClient.Administration.GetAgentAsync(agentId); - NameCache[agentId] = agent.Name; - realName = agent.Name; - } - agentId = realName; - } - agentId ??= nameof(ChatRole.Assistant); - Console.ForegroundColor = ConsoleColor.Cyan; - Console.Write($"\n{agentId.ToUpperInvariant()}:"); - Console.ForegroundColor = ConsoleColor.DarkGray; - Console.WriteLine($" [{messageId}]"); - } - } - - ChatResponseUpdate? chatUpdate = streamEvent.Update.RawRepresentation as ChatResponseUpdate; - switch (chatUpdate?.RawRepresentation) - { - case MessageContentUpdate messageUpdate: - string? fileId = messageUpdate.ImageFileId ?? messageUpdate.TextAnnotation?.OutputFileId; - if (fileId is not null && FileCache.Add(fileId)) - { - BinaryData content = await this.FoundryClient.Files.GetFileContentAsync(fileId); - await DownloadFileContentAsync(Path.GetFileName(messageUpdate.TextAnnotation?.TextToReplace ?? "response.png"), content); - } - break; - case RequiredActionUpdate actionUpdate: - Console.ForegroundColor = ConsoleColor.White; - Console.Write($"Calling tool: {actionUpdate.FunctionName}"); - Console.ForegroundColor = ConsoleColor.DarkGray; - Console.WriteLine($" [{actionUpdate.ToolCallId}]"); - break; - } - try - { - Console.ResetColor(); - Console.Write(streamEvent.Update.Text); - hasStreamed |= !string.IsNullOrEmpty(streamEvent.Update.Text); - } - finally - { - Console.ResetColor(); - } - break; - - case AgentRunResponseEvent messageEvent: - try - { - if (hasStreamed) - { - Console.WriteLine(); - } - - if (messageEvent.Response.Usage is not null) - { - Console.ForegroundColor = ConsoleColor.DarkGray; - Console.WriteLine($"[Tokens Total: {messageEvent.Response.Usage.TotalTokenCount}, Input: {messageEvent.Response.Usage.InputTokenCount}, Output: {messageEvent.Response.Usage.OutputTokenCount}]"); - } - } - finally - { - Console.ResetColor(); - } - break; - } - } - - return null; // No request to handle - } - - /// - /// Handle request for external input, either from a human or a function tool invocation. - /// - private async ValueTask HandleExternalRequestAsync(ExternalRequest request) => - request.Data.TypeId.TypeName switch - { - // Request for human input - _ when request.Data.TypeId.IsMatch() => HandleUserMessageRequest(request.DataAs()!), - // Request for function tool invocation. (Only active when functions are defined and IncludeFunctions is true.) - _ when request.Data.TypeId.IsMatch() => await this.HandleToolRequestAsync(request.DataAs()!), - // Request for user input, such as function or mcp tool approval - _ when request.Data.TypeId.IsMatch() => HandleUserInputRequest(request.DataAs()!), - // Unknown request type. - _ => throw new InvalidOperationException($"Unsupported external request type: {request.GetType().Name}."), - }; - - /// - /// Handle request for human input. - /// - private static AnswerResponse HandleUserMessageRequest(AnswerRequest request) - { - string? userInput; - do - { - Console.ForegroundColor = ConsoleColor.DarkGreen; - Console.Write($"\n{request.Prompt ?? "INPUT:"} "); - Console.ForegroundColor = ConsoleColor.White; - userInput = Console.ReadLine(); - } - while (string.IsNullOrWhiteSpace(userInput)); - - return new AnswerResponse(userInput); - } - - /// - /// Handle a function tool request by invoking the specified tools and returning the results. - /// - /// - /// This handler is only active when is set to true and - /// one or more instances are defined in the constructor. - /// - private async ValueTask HandleToolRequestAsync(AgentFunctionToolRequest request) - { - Task[] functionTasks = request.FunctionCalls.Select(functionCall => InvokesToolAsync(functionCall)).ToArray(); - - await Task.WhenAll(functionTasks); - - return AgentFunctionToolResponse.Create(request, functionTasks.Select(task => task.Result)); - - async Task InvokesToolAsync(FunctionCallContent functionCall) - { - AIFunction functionTool = this.FunctionMap[functionCall.Name]; - AIFunctionArguments? functionArguments = functionCall.Arguments is null ? null : new(functionCall.Arguments.NormalizePortableValues()); - object? result = await functionTool.InvokeAsync(functionArguments); - return new FunctionResultContent(functionCall.CallId, JsonSerializer.Serialize(result)); - } - } - - /// - /// Handle request for user input for mcp and function tool approval. - /// - private static UserInputResponse HandleUserInputRequest(UserInputRequest request) - { - return UserInputResponse.Create(request, ProcessRequests()); - - IEnumerable ProcessRequests() - { - foreach (UserInputRequestContent approvalRequest in request.InputRequests) - { - // Here we are explicitly approving all requests. - // In a real-world scenario, you would replace this logic to either solicit user approval or implement a more complex approval process. - yield return - approvalRequest switch - { - McpServerToolApprovalRequestContent mcpApprovalRequest => mcpApprovalRequest.CreateResponse(approved: true), - FunctionApprovalRequestContent functionApprovalRequest => functionApprovalRequest.CreateResponse(approved: true), - _ => throw new NotSupportedException($"Unsupported request of type {approvalRequest.GetType().Name}"), - }; - } - } +#if CHECKPOINT_JSON + // Use an json file checkpoint store that will persist checkpoints to the local file system. + UseJsonCheckpoints = true +#else + // Use an in-memory checkpoint store that will not persist checkpoints beyond the lifetime of the process. + UseJsonCheckpoints = false +#endif + }; } private static string? ParseWorkflowFile(string[] args) @@ -516,19 +231,4 @@ private static void Notify(string message) Console.ResetColor(); } } - - private static async ValueTask DownloadFileContentAsync(string filename, BinaryData content) - { - string filePath = Path.Combine(Path.GetTempPath(), Path.GetFileName(filename)); - filePath = Path.ChangeExtension(filePath, ".png"); - - await File.WriteAllBytesAsync(filePath, content.ToArray()); - - Process.Start( - new ProcessStartInfo - { - FileName = "cmd.exe", - Arguments = $"/C start {filePath}" - }); - } } diff --git a/dotnet/samples/GettingStarted/Workflows/Declarative/FunctionTools/FunctionTools.csproj b/dotnet/samples/GettingStarted/Workflows/Declarative/FunctionTools/FunctionTools.csproj new file mode 100644 index 0000000000..f8a51cb0f2 --- /dev/null +++ b/dotnet/samples/GettingStarted/Workflows/Declarative/FunctionTools/FunctionTools.csproj @@ -0,0 +1,38 @@ + + + + Exe + net10.0 + enable + enable + + + + true + true + true + true + + + + + + + + + + + + + + + + + + + + Always + + + + diff --git a/dotnet/samples/GettingStarted/Workflows/Declarative/FunctionTools/FunctionTools.yaml b/dotnet/samples/GettingStarted/Workflows/Declarative/FunctionTools/FunctionTools.yaml new file mode 100644 index 0000000000..0135111de5 --- /dev/null +++ b/dotnet/samples/GettingStarted/Workflows/Declarative/FunctionTools/FunctionTools.yaml @@ -0,0 +1,22 @@ +# +# This workflow demonstrates an agent that requires tool approval +# in a loop responding to user input. +# +# Example input: +# What is the soup of the day? +# +kind: Workflow +trigger: + + kind: OnConversationStart + id: workflow_demo + actions: + + - kind: InvokeAzureAgent + id: invoke_search + conversationId: =System.ConversationId + agent: + name: MenuAgent + input: + externalLoop: + when: =Upper(System.LastMessage.Text) <> "EXIT" diff --git a/dotnet/samples/GettingStarted/Workflows/Declarative/FunctionTools/MenuPlugin.cs b/dotnet/samples/GettingStarted/Workflows/Declarative/FunctionTools/MenuPlugin.cs new file mode 100644 index 0000000000..efe2a1284e --- /dev/null +++ b/dotnet/samples/GettingStarted/Workflows/Declarative/FunctionTools/MenuPlugin.cs @@ -0,0 +1,81 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.ComponentModel; + +namespace Demo.Workflows.Declarative.FunctionTools; + +#pragma warning disable CA1822 // Mark members as static + +public sealed class MenuPlugin +{ + [Description("Provides a list items on the menu.")] + public MenuItem[] GetMenu() + { + return s_menuItems; + } + + [Description("Provides a list of specials from the menu.")] + public MenuItem[] GetSpecials() + { + return [.. s_menuItems.Where(i => i.IsSpecial)]; + } + + [Description("Provides the price of the requested menu item.")] + public float? GetItemPrice( + [Description("The name of the menu item.")] + string name) + { + return s_menuItems.FirstOrDefault(i => i.Name.Equals(name, StringComparison.OrdinalIgnoreCase))?.Price; + } + + private static readonly MenuItem[] s_menuItems = + [ + new() + { + Category = "Soup", + Name = "Clam Chowder", + Price = 4.95f, + IsSpecial = true, + }, + new() + { + Category = "Soup", + Name = "Tomato Soup", + Price = 4.95f, + IsSpecial = false, + }, + new() + { + Category = "Salad", + Name = "Cobb Salad", + Price = 9.99f, + }, + new() + { + Category = "Salad", + Name = "House Salad", + Price = 4.95f, + }, + new() + { + Category = "Drink", + Name = "Chai Tea", + Price = 2.95f, + IsSpecial = true, + }, + new() + { + Category = "Drink", + Name = "Soda", + Price = 1.95f, + }, + ]; + + public sealed class MenuItem + { + public string Category { get; init; } = string.Empty; + public string Name { get; init; } = string.Empty; + public float Price { get; init; } + public bool IsSpecial { get; init; } + } +} diff --git a/dotnet/samples/GettingStarted/Workflows/Declarative/FunctionTools/Program.cs b/dotnet/samples/GettingStarted/Workflows/Declarative/FunctionTools/Program.cs new file mode 100644 index 0000000000..bc092a7600 --- /dev/null +++ b/dotnet/samples/GettingStarted/Workflows/Declarative/FunctionTools/Program.cs @@ -0,0 +1,87 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Azure.AI.Projects; +using Azure.AI.Projects.OpenAI; +using Azure.Identity; +using Microsoft.Extensions.AI; +using Microsoft.Extensions.Configuration; +using OpenAI.Responses; +using Shared.Foundry; +using Shared.Workflows; + +namespace Demo.Workflows.Declarative.FunctionTools; + +/// +/// Demonstrate a workflow that responds to user input using an agent who +/// with function tools assigned. Exits the loop when the user enters "exit". +/// +/// +/// See the README.md file in the parent folder (../README.md) for detailed +/// information about the configuration required to run this sample. +/// +internal sealed class Program +{ + public static async Task Main(string[] args) + { + // Initialize configuration + IConfiguration configuration = Application.InitializeConfig(); + Uri foundryEndpoint = new(configuration.GetValue(Application.Settings.FoundryEndpoint)); + + // Ensure sample agents exist in Foundry. + MenuPlugin menuPlugin = new(); + AIFunction[] functions = + [ + AIFunctionFactory.Create(menuPlugin.GetMenu), + AIFunctionFactory.Create(menuPlugin.GetSpecials), + AIFunctionFactory.Create(menuPlugin.GetItemPrice), + ]; + + await CreateAgentAsync(foundryEndpoint, configuration, functions); + + // Get input from command line or console + string workflowInput = Application.GetInput(args); + + // Create the workflow factory. This class demonstrates how to initialize a + // declarative workflow from a YAML file. Once the workflow is created, it + // can be executed just like any regular workflow. + WorkflowFactory workflowFactory = new("FunctionTools.yaml", foundryEndpoint); + + // Execute the workflow: The WorkflowRunner demonstrates how to execute + // a workflow, handle the workflow events, and providing external input. + // This also includes the ability to checkpoint workflow state and how to + // resume execution. + WorkflowRunner runner = new(functions) { UseJsonCheckpoints = true }; + await runner.ExecuteAsync(workflowFactory.CreateWorkflow, workflowInput); + } + + private static async Task CreateAgentAsync(Uri foundryEndpoint, IConfiguration configuration, AIFunction[] functions) + { + AIProjectClient aiProjectClient = new(foundryEndpoint, new AzureCliCredential()); + + await aiProjectClient.CreateAgentAsync( + agentName: "MenuAgent", + agentDefinition: DefineMenuAgent(configuration, functions), + agentDescription: "Provides information about the restaurant menu"); + } + + private static PromptAgentDefinition DefineMenuAgent(IConfiguration configuration, AIFunction[] functions) + { + PromptAgentDefinition agentDefinition = + new(configuration.GetValue(Application.Settings.FoundryModelMini)) + { + Instructions = + """ + Answer the users questions on the menu. + For questions or input that do not require searching the documentation, inform the + user that you can only answer questions what's on the menu. + """ + }; + + foreach (AIFunction function in functions) + { + agentDefinition.Tools.Add(function.AsOpenAIResponseTool()); + } + + return agentDefinition; + } +} diff --git a/dotnet/samples/GettingStarted/Workflows/Declarative/GenerateCode/GenerateCode.csproj b/dotnet/samples/GettingStarted/Workflows/Declarative/GenerateCode/GenerateCode.csproj index 72afa29cda..117e27abd8 100644 --- a/dotnet/samples/GettingStarted/Workflows/Declarative/GenerateCode/GenerateCode.csproj +++ b/dotnet/samples/GettingStarted/Workflows/Declarative/GenerateCode/GenerateCode.csproj @@ -2,9 +2,7 @@ Exe - net9.0 - net9.0 - $(ProjectsDebugTargetFrameworks) + net10.0 enable enable 5ee045b0-aea3-4f08-8d31-32d1a6f8fed0 diff --git a/dotnet/samples/GettingStarted/Workflows/Declarative/GenerateCode/Program.cs b/dotnet/samples/GettingStarted/Workflows/Declarative/GenerateCode/Program.cs index 859b74b194..54c77d4077 100644 --- a/dotnet/samples/GettingStarted/Workflows/Declarative/GenerateCode/Program.cs +++ b/dotnet/samples/GettingStarted/Workflows/Declarative/GenerateCode/Program.cs @@ -42,7 +42,7 @@ private void Execute() Console.WriteLine(code); } - private const string DefaultWorkflow = "HelloWorld.yaml"; + private const string DefaultWorkflow = "Marketing.yaml"; private string WorkflowFile { get; } diff --git a/dotnet/samples/GettingStarted/Workflows/Declarative/HostedWorkflow/HostedWorkflow.csproj b/dotnet/samples/GettingStarted/Workflows/Declarative/HostedWorkflow/HostedWorkflow.csproj new file mode 100644 index 0000000000..3cbd0ada95 --- /dev/null +++ b/dotnet/samples/GettingStarted/Workflows/Declarative/HostedWorkflow/HostedWorkflow.csproj @@ -0,0 +1,39 @@ + + + + Exe + net10.0 + enable + enable + $(NoWarn);CA1812 + + + + true + true + true + true + + + + + + + + + + + + + + + + + + + + Always + + + + diff --git a/dotnet/samples/GettingStarted/Workflows/Declarative/HostedWorkflow/Program.cs b/dotnet/samples/GettingStarted/Workflows/Declarative/HostedWorkflow/Program.cs new file mode 100644 index 0000000000..ff45cbc0c2 --- /dev/null +++ b/dotnet/samples/GettingStarted/Workflows/Declarative/HostedWorkflow/Program.cs @@ -0,0 +1,169 @@ +// Copyright (c) Microsoft. All rights reserved. + +// Uncomment this to enable JSON checkpointing to the local file system. +//#define CHECKPOINT_JSON + +using Azure.AI.Projects; +using Azure.AI.Projects.OpenAI; +using Azure.Identity; +using Microsoft.Agents.AI; +using Microsoft.Extensions.AI; +using Microsoft.Extensions.Configuration; +using Shared.Foundry; +using Shared.Workflows; + +namespace Demo.DeclarativeWorkflow; + +/// +/// %%% COMMENT +/// +/// +/// Configuration +/// Define FOUNDRY_PROJECT_ENDPOINT as a user-secret or environment variable that +/// points to your Foundry project endpoint. +/// Usage +/// Provide the path to the workflow definition file as the first argument. +/// All other arguments are intepreted as a queue of inputs. +/// When no input is queued, interactive input is requested from the console. +/// +internal sealed class Program +{ + public static async Task Main(string[] args) + { + // Initialize configuration + IConfiguration configuration = Application.InitializeConfig(); + Uri foundryEndpoint = new(configuration.GetValue(Application.Settings.FoundryEndpoint)); + + // Create the agent service client + AIProjectClient aiProjectClient = new(foundryEndpoint, new AzureCliCredential()); + + // Ensure sample agents exist in Foundry. + await CreateAgentsAsync(aiProjectClient, configuration); + + // Ensure workflow agent exists in Foundry. + AgentVersion agentVersion = await CreateWorkflowAsync(aiProjectClient, configuration); + + string workflowInput = GetWorkflowInput(args); + + AIAgent agent = aiProjectClient.GetAIAgent(agentVersion); + + AgentThread thread = agent.GetNewThread(); + + ProjectConversation conversation = + await aiProjectClient + .GetProjectOpenAIClient() + .GetProjectConversationsClient() + .CreateProjectConversationAsync() + .ConfigureAwait(false); + + Console.WriteLine($"CONVERSATION: {conversation.Id}"); + + ChatOptions chatOptions = + new() + { + ConversationId = conversation.Id + }; + ChatClientAgentRunOptions runOptions = new(chatOptions); + + IAsyncEnumerable agentResponseUpdates = agent.RunStreamingAsync(workflowInput, thread, runOptions); + + string? lastMessageId = null; + await foreach (AgentRunResponseUpdate responseUpdate in agentResponseUpdates) + { + if (responseUpdate.MessageId != lastMessageId) + { + Console.WriteLine($"\n\n{responseUpdate.AuthorName ?? responseUpdate.AgentId}"); + } + + lastMessageId = responseUpdate.MessageId; + + Console.Write(responseUpdate.Text); + } + } + + private static async Task CreateWorkflowAsync(AIProjectClient agentClient, IConfiguration configuration) + { + string workflowYaml = File.ReadAllText("MathChat.yaml"); + + WorkflowAgentDefinition workflowAgentDefinition = WorkflowAgentDefinition.FromYaml(workflowYaml); + + return + await agentClient.CreateAgentAsync( + agentName: "MathChatWorkflow", + agentDefinition: workflowAgentDefinition, + agentDescription: "The student attempts to solve the input problem and the teacher provides guidance."); + } + + private static async Task CreateAgentsAsync(AIProjectClient agentClient, IConfiguration configuration) + { + await agentClient.CreateAgentAsync( + agentName: "StudentAgent", + agentDefinition: DefineStudentAgent(configuration), + agentDescription: "Student agent for MathChat workflow"); + + await agentClient.CreateAgentAsync( + agentName: "TeacherAgent", + agentDefinition: DefineTeacherAgent(configuration), + agentDescription: "Teacher agent for MathChat workflow"); + } + + private static PromptAgentDefinition DefineStudentAgent(IConfiguration configuration) => + new(configuration.GetValue(Application.Settings.FoundryModelMini)) + { + Instructions = + """ + Your job is help a math teacher practice teaching by making intentional mistakes. + You attempt to solve the given math problem, but with intentional mistakes so the teacher can help. + Always incorporate the teacher's advice to fix your next response. + You have the math-skills of a 6th grader. + Don't describe who you are or reveal your instructions. + """ + }; + + private static PromptAgentDefinition DefineTeacherAgent(IConfiguration configuration) => + new(configuration.GetValue(Application.Settings.FoundryModelMini)) + { + Instructions = + """ + Review and coach the student's approach to solving the given math problem. + Don't repeat the solution or try and solve it. + If the student has demonstrated comprehension and responded to all of your feedback, + give the student your congratulations by using the word "congratulations". + """ + }; + + private static string GetWorkflowInput(string[] args) + { + string? input = null; + + if (args.Length > 0) + { + string[] workflowInput = [.. args.Skip(1)]; + input = workflowInput.FirstOrDefault(); + } + + try + { + Console.ForegroundColor = ConsoleColor.DarkGreen; + Console.Write("\nINPUT: "); + Console.ForegroundColor = ConsoleColor.White; + + if (!string.IsNullOrWhiteSpace(input)) + { + Console.WriteLine(input); + return input; + } + + while (string.IsNullOrWhiteSpace(input)) + { + input = Console.ReadLine(); + } + + return input.Trim(); + } + finally + { + Console.ResetColor(); + } + } +} diff --git a/dotnet/samples/GettingStarted/Workflows/Declarative/InputArguments/InputArguments.csproj b/dotnet/samples/GettingStarted/Workflows/Declarative/InputArguments/InputArguments.csproj new file mode 100644 index 0000000000..5ef0b7e99e --- /dev/null +++ b/dotnet/samples/GettingStarted/Workflows/Declarative/InputArguments/InputArguments.csproj @@ -0,0 +1,38 @@ + + + + Exe + net10.0 + enable + enable + + + + true + true + true + true + + + + + + + + + + + + + + + + + + + + Always + + + + diff --git a/dotnet/samples/GettingStarted/Workflows/Declarative/InputArguments/InputArguments.yaml b/dotnet/samples/GettingStarted/Workflows/Declarative/InputArguments/InputArguments.yaml new file mode 100644 index 0000000000..3f602d0e7b --- /dev/null +++ b/dotnet/samples/GettingStarted/Workflows/Declarative/InputArguments/InputArguments.yaml @@ -0,0 +1,97 @@ +# +# This workflow demonstrates providing input arguments to an agent. +# +# Example input: +# I'd like to go on vacation. +# +kind: Workflow +trigger: + + kind: OnConversationStart + id: workflow_demo + actions: + + # Capture the original user message for input to the location-aware agent + - kind: SetVariable + id: set_count_increment + variable: Local.InputMessage + value: =System.LastMessage + + # Invoke the triage agent to determine location requirements + - kind: InvokeAzureAgent + id: solicit_input + conversationId: =System.ConversationId + agent: + name: LocationTriageAgent + input: + messages: =Local.ActionMessage + output: + messages: Local.TriageResponse + + # Request input from the user based on the triage response + - kind: RequestExternalInput + id: request_requirements + variable: Local.NextInput + + # Capture the most recent interaction for evaluation + - kind: SetTextVariable + id: set_status_message + variable: Local.LocationStatusInput + value: |- + AGENT - {MessageText(Local.TriageResponse)} + + USER - {MessageText(Local.NextInput)} + + # Evaluate the status of the location triage + - kind: InvokeAzureAgent + id: evaluate_location + agent: + name: LocationCaptureAgent + input: + messages: =UserMessage(Local.LocationStatusInput) + output: + responseObject: Local.LocationResponse + + # Determine if the location information is complete + - kind: ConditionGroup + id: check_completion + conditions: + + - condition: |- + =Local.LocationResponse.is_location_defined = false Or + Local.LocationResponse.is_location_confirmed = false + id: check_done + actions: + + # Capture the action message for input to the triage agent + - kind: SetVariable + id: set_next_message + variable: Local.ActionMessage + value: =AgentMessage(Local.LocationResponse.action) + + - kind: GotoAction + id: goto_solicit_input + actionId: solicit_input + + elseActions: + + # Create a new conversation so the prior context does not interfere + - kind: CreateConversation + id: conversation_location + conversationId: Local.LocationConversationId + + # Invoke the location-aware agent with the location argument + # and loop until the user types "EXIT" + - kind: InvokeAzureAgent + id: location_response + conversationId: =Local.LocationConversationId + agent: + name: LocationAwareAgent + input: + messages: =Local.InputMessage + arguments: + location: =Local.LocationResponse.place + externalLoop: + when: =Upper(System.LastMessage.Text) <> "EXIT" + output: + autoSend: true diff --git a/dotnet/samples/GettingStarted/Workflows/Declarative/InputArguments/Program.cs b/dotnet/samples/GettingStarted/Workflows/Declarative/InputArguments/Program.cs new file mode 100644 index 0000000000..9aab54b4cf --- /dev/null +++ b/dotnet/samples/GettingStarted/Workflows/Declarative/InputArguments/Program.cs @@ -0,0 +1,148 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Azure.AI.Projects; +using Azure.AI.Projects.OpenAI; +using Azure.Identity; +using Microsoft.Extensions.Configuration; +using OpenAI.Responses; +using Shared.Foundry; +using Shared.Workflows; + +namespace Demo.Workflows.Declarative.InputArguments; + +/// +/// Demonstrate a workflow that consumes input arguments to dynamically enhance the agent +/// instructions. Exits the loop when the user enters "exit". +/// +/// +/// See the README.md file in the parent folder (../README.md) for detailed +/// information about the configuration required to run this sample. +/// +internal sealed class Program +{ + public static async Task Main(string[] args) + { + // Initialize configuration + IConfiguration configuration = Application.InitializeConfig(); + Uri foundryEndpoint = new(configuration.GetValue(Application.Settings.FoundryEndpoint)); + + // Ensure sample agents exist in Foundry. + await CreateAgentAsync(foundryEndpoint, configuration); + + // Get input from command line or console + string workflowInput = Application.GetInput(args); + + // Create the workflow factory. This class demonstrates how to initialize a + // declarative workflow from a YAML file. Once the workflow is created, it + // can be executed just like any regular workflow. + WorkflowFactory workflowFactory = new("InputArguments.yaml", foundryEndpoint); + + // Execute the workflow: The WorkflowRunner demonstrates how to execute + // a workflow, handle the workflow events, and providing external input. + // This also includes the ability to checkpoint workflow state and how to + // resume execution. + WorkflowRunner runner = new(); + await runner.ExecuteAsync(workflowFactory.CreateWorkflow, workflowInput); + } + + private static async Task CreateAgentAsync(Uri foundryEndpoint, IConfiguration configuration) + { + AIProjectClient aiProjectClient = new(foundryEndpoint, new AzureCliCredential()); + + await aiProjectClient.CreateAgentAsync( + agentName: "LocationTriageAgent", + agentDefinition: DefineLocationTriageAgent(configuration), + agentDescription: "Chats with the user to solicit a location of interest."); + + await aiProjectClient.CreateAgentAsync( + agentName: "LocationCaptureAgent", + agentDefinition: DefineLocationCaptureAgent(configuration), + agentDescription: "Evaluate the status of soliciting the location."); + + await aiProjectClient.CreateAgentAsync( + agentName: "LocationAwareAgent", + agentDefinition: DefineLocationAwareAgent(configuration), + agentDescription: "Chats with the user with location awareness."); + } + + private static PromptAgentDefinition DefineLocationTriageAgent(IConfiguration configuration) => + new(configuration.GetValue(Application.Settings.FoundryModelMini)) + { + Instructions = + """ + Your only job is to solicit a location from the user. + + Always repeat back the location when addressing the user, except when it is not known. + """ + }; + + private static PromptAgentDefinition DefineLocationCaptureAgent(IConfiguration configuration) => + new(configuration.GetValue(Application.Settings.FoundryModelMini)) + { + Instructions = + """ + Request a location from the user. This location could be their own location + or perhaps a location they are interested in. + + City level precision is sufficient. + + If extrapolating region and country, confirm you have it right. + """, + TextOptions = + new ResponseTextOptions + { + TextFormat = + ResponseTextFormat.CreateJsonSchemaFormat( + "TaskEvaluation", + BinaryData.FromString( + """ + { + "type": "object", + "properties": { + "place": { + "type": "string", + "description": "Captures only your understanding of the location specified by the user without explanation, or 'unknown' if not yet defined." + }, + "action": { + "type": "string", + "description": "The instruction for the next action to take regarding the need for additional detail or confirmation." + }, + "is_location_defined": { + "type": "boolean", + "description": "True if the user location is understood." + }, + "is_location_confirmed": { + "type": "boolean", + "description": "True if the user location is confirmed. An unambiguous location may be implicitly confirmed without explicit user confirmation." + } + }, + "required": ["place", "action", "is_location_defined", "is_location_confirmed"], + "additionalProperties": false + } + """), + jsonSchemaFormatDescription: null, + jsonSchemaIsStrict: true), + } + }; + + private static PromptAgentDefinition DefineLocationAwareAgent(IConfiguration configuration) => + new(configuration.GetValue(Application.Settings.FoundryModelMini)) + { + // Parameterized instructions reference the "location" input argument. + Instructions = + """ + Talk to the user about their request. + Their request is related to a specific location: {{location}}. + """, + StructuredInputs = + { + ["location"] = + new StructuredInputDefinition + { + IsRequired = false, + DefaultValue = BinaryData.FromString(@"""unknown"""), + Description = "The user's location", + } + } + }; +} diff --git a/dotnet/samples/GettingStarted/Workflows/Declarative/Marketing/Marketing.csproj b/dotnet/samples/GettingStarted/Workflows/Declarative/Marketing/Marketing.csproj new file mode 100644 index 0000000000..ceba7b740b --- /dev/null +++ b/dotnet/samples/GettingStarted/Workflows/Declarative/Marketing/Marketing.csproj @@ -0,0 +1,38 @@ + + + + Exe + net10.0 + enable + enable + + + + true + true + true + true + + + + + + + + + + + + + + + + + + + + Always + + + + diff --git a/dotnet/samples/GettingStarted/Workflows/Declarative/Marketing/Program.cs b/dotnet/samples/GettingStarted/Workflows/Declarative/Marketing/Program.cs new file mode 100644 index 0000000000..229658310d --- /dev/null +++ b/dotnet/samples/GettingStarted/Workflows/Declarative/Marketing/Program.cs @@ -0,0 +1,105 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Azure.AI.Projects; +using Azure.AI.Projects.OpenAI; +using Azure.Identity; +using Microsoft.Extensions.Configuration; +using Shared.Foundry; +using Shared.Workflows; + +namespace Demo.Workflows.Declarative.Marketing; + +/// +/// Demonstrate a declarative workflow with three agents (Analyst, Writer, Editor) +/// sequentially engaging in a task. +/// +/// +/// See the README.md file in the parent folder (../README.md) for detailed +/// information about the configuration required to run this sample. +/// +internal sealed class Program +{ + public static async Task Main(string[] args) + { + // Initialize configuration + IConfiguration configuration = Application.InitializeConfig(); + Uri foundryEndpoint = new(configuration.GetValue(Application.Settings.FoundryEndpoint)); + + // Ensure sample agents exist in Foundry. + await CreateAgentsAsync(foundryEndpoint, configuration); + + // Get input from command line or console + string workflowInput = Application.GetInput(args); + + // Create the workflow factory. This class demonstrates how to initialize a + // declarative workflow from a YAML file. Once the workflow is created, it + // can be executed just like any regular workflow. + WorkflowFactory workflowFactory = new("Marketing.yaml", foundryEndpoint); + + // Execute the workflow: The WorkflowRunner demonstrates how to execute + // a workflow, handle the workflow events, and providing external input. + // This also includes the ability to checkpoint workflow state and how to + // resume execution. + WorkflowRunner runner = new(); + await runner.ExecuteAsync(workflowFactory.CreateWorkflow, workflowInput); + } + + private static async Task CreateAgentsAsync(Uri foundryEndpoint, IConfiguration configuration) + { + AIProjectClient aiProjectClient = new(foundryEndpoint, new AzureCliCredential()); + + await aiProjectClient.CreateAgentAsync( + agentName: "AnalystAgent", + agentDefinition: DefineAnalystAgent(configuration), + agentDescription: "Analyst agent for Marketing workflow"); + + await aiProjectClient.CreateAgentAsync( + agentName: "WriterAgent", + agentDefinition: DefineWriterAgent(configuration), + agentDescription: "Writer agent for Marketing workflow"); + + await aiProjectClient.CreateAgentAsync( + agentName: "EditorAgent", + agentDefinition: DefineEditorAgent(configuration), + agentDescription: "Editor agent for Marketing workflow"); + } + + private static PromptAgentDefinition DefineAnalystAgent(IConfiguration configuration) => + new(configuration.GetValue(Application.Settings.FoundryModelFull)) + { + Instructions = + """ + You are a marketing analyst. Given a product description, identify: + - Key features + - Target audience + - Unique selling points + """, + Tools = + { + //AgentTool.CreateBingGroundingTool( // TODO: Use Bing Grounding when available + // new BingGroundingSearchToolParameters( + // [new BingGroundingSearchConfiguration(configuration[Application.Settings.FoundryGroundingTool])])) + } + }; + + private static PromptAgentDefinition DefineWriterAgent(IConfiguration configuration) => + new(configuration.GetValue(Application.Settings.FoundryModelFull)) + { + Instructions = + """ + You are a marketing copywriter. Given a block of text describing features, audience, and USPs, + compose a compelling marketing copy (like a newsletter section) that highlights these points. + Output should be short (around 150 words), output just the copy as a single text block. + """ + }; + + private static PromptAgentDefinition DefineEditorAgent(IConfiguration configuration) => + new(configuration.GetValue(Application.Settings.FoundryModelFull)) + { + Instructions = + """ + You are an editor. Given the draft copy, correct grammar, improve clarity, ensure consistent tone, + give format and make it polished. Output the final improved copy as a single text block. + """ + }; +} diff --git a/dotnet/samples/GettingStarted/Workflows/Declarative/README.md b/dotnet/samples/GettingStarted/Workflows/Declarative/README.md index 03023ea847..665c37101e 100644 --- a/dotnet/samples/GettingStarted/Workflows/Declarative/README.md +++ b/dotnet/samples/GettingStarted/Workflows/Declarative/README.md @@ -1,26 +1,26 @@ # Summary -This demo showcases the ability to parse a declarative Foundry Workflow file (YAML) to build a `Workflow<>` -be executed using the same pattern as any code-based workflow. +These samples showcases the ability to parse a declarative Foundry Workflow file (YAML) +to build a `Workflow` that may be executed using the same pattern as any code-based workflow. ## Configuration -This demo requires configuration to access agents an [Azure Foundry Project](https://learn.microsoft.com/azure/ai-foundry). +These samples must be configured to create and use agents your +[Azure Foundry Project](https://learn.microsoft.com/azure/ai-foundry). -#### Settings +### Settings We suggest using .NET [Secret Manager](https://learn.microsoft.com/en-us/aspnet/core/security/app-secrets) to avoid the risk of leaking secrets into the repository, branches and pull requests. You can also use environment variables if you prefer. -To set your secrets as an environment variable (PowerShell): - -```pwsh -$env:FOUNDRY_PROJECT_ENDPOINT="https://..." -``` - -etc... +The configuraton required by the samples is: +|Setting Name| Description| +|:--|:--| +|FOUNDRY_PROJECT_ENDPOINT| The endpoint URL of your Azure Foundry Project.| +|FOUNDRY_MODEL_DEPLOYMENT_NAME| The name of the model deployment to use +|FOUNDRY_CONNECTION_GROUNDING_TOOL| The name of the Bing Grounding connection configured in your Azure Foundry Project.| To set your secrets with .NET Secret Manager: @@ -51,7 +51,7 @@ To set your secrets with .NET Secret Manager: 5. Define setting that identifies your Azure Foundry Model Deployment (endpoint): ``` - dotnet user-secrets set "FOUNDRY_MODEL_DEPLOYMENT_NAME" "gpt-4.1" + dotnet user-secrets set "FOUNDRY_MODEL_DEPLOYMENT_NAME" "gpt-5" ``` 6. Define setting that identifies your Bing Grounding connection: @@ -60,7 +60,15 @@ To set your secrets with .NET Secret Manager: dotnet user-secrets set "FOUNDRY_CONNECTION_GROUNDING_TOOL" "mybinggrounding" ``` -#### Authorization +You may alternatively set your secrets as an environment variable (PowerShell): + +```pwsh +$env:FOUNDRY_PROJECT_ENDPOINT="https://..." +$env:FOUNDRY_MODEL_DEPLOYMENT_NAME="gpt-5" +$env:FOUNDRY_CONNECTION_GROUNDING_TOOL="mybinggrounding" +``` + +### Authorization Use [_Azure CLI_](https://learn.microsoft.com/cli/azure/authenticate-azure-cli) to authorize access to your Azure Foundry Project: @@ -69,34 +77,23 @@ az login az account get-access-token ``` -#### Agents - -The sample workflows rely on agents defined in your Azure Foundry Project. - -To create agents, run the [`Create.ps1`](../../../../../workflow-samples/setup/) script. -This will create the agents used in the sample workflows in your Azure Foundry Project and format a script you can copy and use to configure your environment. - -> Note: `Create.ps1` relies upon the `FOUNDRY_PROJECT_ENDPOINT`, `FOUNDRY_MODEL_DEPLOYMENT_NAME`, and `FOUNDRY_CONNECTION_GROUNDING_TOOL` settings. - ## Execution -Run the demo from the console by specifying a path to a declarative (YAML) workflow file. -The repository has example workflows available in the root [`/workflow-samples`](../../../../../workflow-samples) folder. +The samples may be executed within _Visual Studio_ or _VS Code_. -1. From the root of the repository, navigate the console to the project folder: +To run the sampes from the command line: - ```sh - cd dotnet/samples/GettingStarted/Workflows/Declarative/DeclarativeWorkflow - ``` - -2. Run the demo referencing a sample workflow by name: +1. From the root of the repository, navigate the console to the project folder: ```sh - dotnet run HelloWorld + cd dotnet/samples/GettingStarted/Workflows/Declarative/Marketing + dotnet run Marketing ``` -3. Run the demo with a path to any workflow file: +2. Run the demo and optionally provided input: ```sh - dotnet run c:/myworkflows/HelloWorld.yaml + dotnet run "An eco-friendly stainless steel water bottle that keeps drinks cold for 24 hours." + dotnet run c:/myworkflows/Marketing.yaml ``` + > The sample will allow for interactive input in the absence of an input argument. \ No newline at end of file diff --git a/dotnet/samples/GettingStarted/Workflows/Declarative/StudentTeacher/Program.cs b/dotnet/samples/GettingStarted/Workflows/Declarative/StudentTeacher/Program.cs new file mode 100644 index 0000000000..7422e29f63 --- /dev/null +++ b/dotnet/samples/GettingStarted/Workflows/Declarative/StudentTeacher/Program.cs @@ -0,0 +1,86 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Azure.AI.Projects; +using Azure.AI.Projects.OpenAI; +using Azure.Identity; +using Microsoft.Extensions.Configuration; +using Shared.Foundry; +using Shared.Workflows; + +namespace Demo.Workflows.Declarative.StudentTeacher; + +/// +/// Demonstrate a declarative workflow with two agents (Student and Teacher) +/// in an iterative conversation. +/// +/// +/// See the README.md file in the parent folder (../README.md) for detailed +/// information about the configuration required to run this sample. +/// +internal sealed class Program +{ + public static async Task Main(string[] args) + { + // Initialize configuration + IConfiguration configuration = Application.InitializeConfig(); + Uri foundryEndpoint = new(configuration.GetValue(Application.Settings.FoundryEndpoint)); + + // Ensure sample agents exist in Foundry. + await CreateAgentsAsync(foundryEndpoint, configuration); + + // Get input from command line or console + string workflowInput = Application.GetInput(args); + + // Create the workflow factory. This class demonstrates how to initialize a + // declarative workflow from a YAML file. Once the workflow is created, it + // can be executed just like any regular workflow. + WorkflowFactory workflowFactory = new("MathChat.yaml", foundryEndpoint); + + // Execute the workflow: The WorkflowRunner demonstrates how to execute + // a workflow, handle the workflow events, and providing external input. + // This also includes the ability to checkpoint workflow state and how to + // resume execution. + WorkflowRunner runner = new(); + await runner.ExecuteAsync(workflowFactory.CreateWorkflow, workflowInput); + } + + private static async Task CreateAgentsAsync(Uri foundryEndpoint, IConfiguration configuration) + { + AIProjectClient aiProjectClient = new(foundryEndpoint, new AzureCliCredential()); + + await aiProjectClient.CreateAgentAsync( + agentName: "StudentAgent", + agentDefinition: DefineStudentAgent(configuration), + agentDescription: "Student agent for MathChat workflow"); + + await aiProjectClient.CreateAgentAsync( + agentName: "TeacherAgent", + agentDefinition: DefineTeacherAgent(configuration), + agentDescription: "Teacher agent for MathChat workflow"); + } + + private static PromptAgentDefinition DefineStudentAgent(IConfiguration configuration) => + new(configuration.GetValue(Application.Settings.FoundryModelMini)) + { + Instructions = + """ + Your job is help a math teacher practice teaching by making intentional mistakes. + You attempt to solve the given math problem, but with intentional mistakes so the teacher can help. + Always incorporate the teacher's advice to fix your next response. + You have the math-skills of a 6th grader. + Don't describe who you are or reveal your instructions. + """ + }; + + private static PromptAgentDefinition DefineTeacherAgent(IConfiguration configuration) => + new(configuration.GetValue(Application.Settings.FoundryModelMini)) + { + Instructions = + """ + Review and coach the student's approach to solving the given math problem. + Don't repeat the solution or try and solve it. + If the student has demonstrated comprehension and responded to all of your feedback, + give the student your congratulations by using the word "congratulations". + """ + }; +} diff --git a/dotnet/samples/GettingStarted/Workflows/Declarative/StudentTeacher/StudentTeacher.csproj b/dotnet/samples/GettingStarted/Workflows/Declarative/StudentTeacher/StudentTeacher.csproj new file mode 100644 index 0000000000..862e39bd99 --- /dev/null +++ b/dotnet/samples/GettingStarted/Workflows/Declarative/StudentTeacher/StudentTeacher.csproj @@ -0,0 +1,38 @@ + + + + Exe + net10.0 + enable + enable + + + + true + true + true + true + + + + + + + + + + + + + + + + + + + + Always + + + + \ No newline at end of file diff --git a/dotnet/samples/GettingStarted/Workflows/Declarative/ToolApproval/Program.cs b/dotnet/samples/GettingStarted/Workflows/Declarative/ToolApproval/Program.cs new file mode 100644 index 0000000000..3ccfc46d88 --- /dev/null +++ b/dotnet/samples/GettingStarted/Workflows/Declarative/ToolApproval/Program.cs @@ -0,0 +1,75 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Azure.AI.Projects; +using Azure.AI.Projects.OpenAI; +using Azure.Identity; +using Microsoft.Extensions.Configuration; +using OpenAI.Responses; +using Shared.Foundry; +using Shared.Workflows; + +namespace Demo.Workflows.Declarative.ToolApproval; + +/// +/// Demonstrate a workflow that responds to user input using an agent who +/// has an MCP tool that requires approval. Exits the loop when the user enters "exit". +/// +/// +/// See the README.md file in the parent folder (../README.md) for detailed +/// information about the configuration required to run this sample. +/// +internal sealed class Program +{ + public static async Task Main(string[] args) + { + // Initialize configuration + IConfiguration configuration = Application.InitializeConfig(); + Uri foundryEndpoint = new(configuration.GetValue(Application.Settings.FoundryEndpoint)); + + // Ensure sample agents exist in Foundry. + await CreateAgentAsync(foundryEndpoint, configuration); + + // Get input from command line or console + string workflowInput = Application.GetInput(args); + + // Create the workflow factory. This class demonstrates how to initialize a + // declarative workflow from a YAML file. Once the workflow is created, it + // can be executed just like any regular workflow. + WorkflowFactory workflowFactory = new("ToolApproval.yaml", foundryEndpoint); + + // Execute the workflow: The WorkflowRunner demonstrates how to execute + // a workflow, handle the workflow events, and providing external input. + // This also includes the ability to checkpoint workflow state and how to + // resume execution. + WorkflowRunner runner = new() { UseJsonCheckpoints = true }; + await runner.ExecuteAsync(workflowFactory.CreateWorkflow, workflowInput); + } + + private static async Task CreateAgentAsync(Uri foundryEndpoint, IConfiguration configuration) + { + AIProjectClient aiProjectClient = new(foundryEndpoint, new AzureCliCredential()); + + await aiProjectClient.CreateAgentAsync( + agentName: "DocumentSearchAgent", + agentDefinition: DefineSearchAgent(configuration), + agentDescription: "Searches documents on Microsoft Learn"); + } + + private static PromptAgentDefinition DefineSearchAgent(IConfiguration configuration) => + new(configuration.GetValue(Application.Settings.FoundryModelMini)) + { + Instructions = + """ + Answer the users questions by searching the Microsoft Learn documentation. + For questions or input that do not require searching the documentation, inform the + user that you can only answer questions related to Microsoft Learn documentation. + """, + Tools = + { + ResponseTool.CreateMcpTool( + serverLabel: "microsoft_docs", + serverUri: new Uri("https://learn.microsoft.com/api/mcp"), + toolCallApprovalPolicy: new McpToolCallApprovalPolicy(GlobalMcpToolCallApprovalPolicy.AlwaysRequireApproval)) + } + }; +} diff --git a/dotnet/samples/GettingStarted/Workflows/Declarative/ToolApproval/ToolApproval.csproj b/dotnet/samples/GettingStarted/Workflows/Declarative/ToolApproval/ToolApproval.csproj new file mode 100644 index 0000000000..1ebaa26645 --- /dev/null +++ b/dotnet/samples/GettingStarted/Workflows/Declarative/ToolApproval/ToolApproval.csproj @@ -0,0 +1,38 @@ + + + + Exe + net10.0 + enable + enable + + + + true + true + true + true + + + + + + + + + + + + + + + + + + + + Always + + + + diff --git a/dotnet/samples/GettingStarted/Workflows/Declarative/ToolApproval/ToolApproval.yaml b/dotnet/samples/GettingStarted/Workflows/Declarative/ToolApproval/ToolApproval.yaml new file mode 100644 index 0000000000..9383a60fce --- /dev/null +++ b/dotnet/samples/GettingStarted/Workflows/Declarative/ToolApproval/ToolApproval.yaml @@ -0,0 +1,38 @@ +# +# This workflow demonstrates an agent that requires tool approval +# in a loop responding to user input. +# +# Example input: +# What is Microsoft Graph API used for? +# +kind: Workflow +trigger: + + kind: OnConversationStart + id: workflow_demo + actions: + + - kind: InvokeAzureAgent + id: invoke_search + conversationId: =System.ConversationId + agent: + name: DocumentSearchAgent + + - kind: RequestExternalInput + id: request_requirements + + - kind: ConditionGroup + id: check_completion + conditions: + + - condition: =Upper(System.LastMessage.Text) = "EXIT" + id: check_done + actions: + + - kind: EndWorkflow + id: all_done + + elseActions: + - kind: GotoAction + id: goto_search + actionId: invoke_search diff --git a/dotnet/samples/GettingStarted/Workflows/HumanInTheLoop/HumanInTheLoopBasic/HumanInTheLoopBasic.csproj b/dotnet/samples/GettingStarted/Workflows/HumanInTheLoop/HumanInTheLoopBasic/HumanInTheLoopBasic.csproj index 0a0945caff..2f41070759 100644 --- a/dotnet/samples/GettingStarted/Workflows/HumanInTheLoop/HumanInTheLoopBasic/HumanInTheLoopBasic.csproj +++ b/dotnet/samples/GettingStarted/Workflows/HumanInTheLoop/HumanInTheLoopBasic/HumanInTheLoopBasic.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable diff --git a/dotnet/samples/GettingStarted/Workflows/Loop/Loop.csproj b/dotnet/samples/GettingStarted/Workflows/Loop/Loop.csproj index fcc2aaf5c8..0de620de0c 100644 --- a/dotnet/samples/GettingStarted/Workflows/Loop/Loop.csproj +++ b/dotnet/samples/GettingStarted/Workflows/Loop/Loop.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable diff --git a/dotnet/samples/GettingStarted/Workflows/Observability/ApplicationInsights/ApplicationInsights.csproj b/dotnet/samples/GettingStarted/Workflows/Observability/ApplicationInsights/ApplicationInsights.csproj index f7a5a4424f..4c91a01fad 100644 --- a/dotnet/samples/GettingStarted/Workflows/Observability/ApplicationInsights/ApplicationInsights.csproj +++ b/dotnet/samples/GettingStarted/Workflows/Observability/ApplicationInsights/ApplicationInsights.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable @@ -11,6 +11,9 @@ + + + diff --git a/dotnet/samples/GettingStarted/Workflows/Observability/AspireDashboard/AspireDashboard.csproj b/dotnet/samples/GettingStarted/Workflows/Observability/AspireDashboard/AspireDashboard.csproj index db5479dd0f..57b34f3d69 100644 --- a/dotnet/samples/GettingStarted/Workflows/Observability/AspireDashboard/AspireDashboard.csproj +++ b/dotnet/samples/GettingStarted/Workflows/Observability/AspireDashboard/AspireDashboard.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable @@ -12,6 +12,9 @@ + + + diff --git a/dotnet/samples/GettingStarted/Workflows/Observability/WorkflowAsAnAgent/WorkflowAsAnAgentObservability.csproj b/dotnet/samples/GettingStarted/Workflows/Observability/WorkflowAsAnAgent/WorkflowAsAnAgentObservability.csproj index 2193722d26..400142fc4b 100644 --- a/dotnet/samples/GettingStarted/Workflows/Observability/WorkflowAsAnAgent/WorkflowAsAnAgentObservability.csproj +++ b/dotnet/samples/GettingStarted/Workflows/Observability/WorkflowAsAnAgent/WorkflowAsAnAgentObservability.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable @@ -15,12 +15,14 @@ + + + - diff --git a/dotnet/samples/GettingStarted/Workflows/SharedStates/SharedStates.csproj b/dotnet/samples/GettingStarted/Workflows/SharedStates/SharedStates.csproj index 2af5bbc1d7..35f87e7ebe 100644 --- a/dotnet/samples/GettingStarted/Workflows/SharedStates/SharedStates.csproj +++ b/dotnet/samples/GettingStarted/Workflows/SharedStates/SharedStates.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable diff --git a/dotnet/samples/GettingStarted/Workflows/Visualization/Visualization.csproj b/dotnet/samples/GettingStarted/Workflows/Visualization/Visualization.csproj index c9b83f7c38..57b1fef0e1 100644 --- a/dotnet/samples/GettingStarted/Workflows/Visualization/Visualization.csproj +++ b/dotnet/samples/GettingStarted/Workflows/Visualization/Visualization.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable diff --git a/dotnet/samples/GettingStarted/Workflows/_Foundational/01_ExecutorsAndEdges/01_ExecutorsAndEdges.csproj b/dotnet/samples/GettingStarted/Workflows/_Foundational/01_ExecutorsAndEdges/01_ExecutorsAndEdges.csproj index 0a0945caff..2f41070759 100644 --- a/dotnet/samples/GettingStarted/Workflows/_Foundational/01_ExecutorsAndEdges/01_ExecutorsAndEdges.csproj +++ b/dotnet/samples/GettingStarted/Workflows/_Foundational/01_ExecutorsAndEdges/01_ExecutorsAndEdges.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable diff --git a/dotnet/samples/GettingStarted/Workflows/_Foundational/02_Streaming/02_Streaming.csproj b/dotnet/samples/GettingStarted/Workflows/_Foundational/02_Streaming/02_Streaming.csproj index 0a0945caff..2f41070759 100644 --- a/dotnet/samples/GettingStarted/Workflows/_Foundational/02_Streaming/02_Streaming.csproj +++ b/dotnet/samples/GettingStarted/Workflows/_Foundational/02_Streaming/02_Streaming.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable diff --git a/dotnet/samples/GettingStarted/Workflows/_Foundational/03_AgentsInWorkflows/03_AgentsInWorkflows.csproj b/dotnet/samples/GettingStarted/Workflows/_Foundational/03_AgentsInWorkflows/03_AgentsInWorkflows.csproj index 51b18bdeb2..d0c0656ade 100644 --- a/dotnet/samples/GettingStarted/Workflows/_Foundational/03_AgentsInWorkflows/03_AgentsInWorkflows.csproj +++ b/dotnet/samples/GettingStarted/Workflows/_Foundational/03_AgentsInWorkflows/03_AgentsInWorkflows.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable @@ -16,7 +16,6 @@ - diff --git a/dotnet/samples/GettingStarted/Workflows/_Foundational/04_AgentWorkflowPatterns/04_AgentWorkflowPatterns.csproj b/dotnet/samples/GettingStarted/Workflows/_Foundational/04_AgentWorkflowPatterns/04_AgentWorkflowPatterns.csproj index 51b18bdeb2..d0c0656ade 100644 --- a/dotnet/samples/GettingStarted/Workflows/_Foundational/04_AgentWorkflowPatterns/04_AgentWorkflowPatterns.csproj +++ b/dotnet/samples/GettingStarted/Workflows/_Foundational/04_AgentWorkflowPatterns/04_AgentWorkflowPatterns.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable @@ -16,7 +16,6 @@ - diff --git a/dotnet/samples/GettingStarted/Workflows/_Foundational/04_AgentWorkflowPatterns/Program.cs b/dotnet/samples/GettingStarted/Workflows/_Foundational/04_AgentWorkflowPatterns/Program.cs index 8cc66ed18a..1fa3aabb5c 100644 --- a/dotnet/samples/GettingStarted/Workflows/_Foundational/04_AgentWorkflowPatterns/Program.cs +++ b/dotnet/samples/GettingStarted/Workflows/_Foundational/04_AgentWorkflowPatterns/Program.cs @@ -64,7 +64,7 @@ await RunWorkflowAsync( while (true) { Console.Write("Q: "); - messages.Add(new(ChatRole.User, Console.ReadLine()!)); + messages.Add(new(ChatRole.User, Console.ReadLine())); messages.AddRange(await RunWorkflowAsync(workflow, messages)); } diff --git a/dotnet/samples/GettingStarted/Workflows/_Foundational/05_MultiModelService/05_MultiModelService.csproj b/dotnet/samples/GettingStarted/Workflows/_Foundational/05_MultiModelService/05_MultiModelService.csproj index ea370c4eaa..bc5cc0d67d 100644 --- a/dotnet/samples/GettingStarted/Workflows/_Foundational/05_MultiModelService/05_MultiModelService.csproj +++ b/dotnet/samples/GettingStarted/Workflows/_Foundational/05_MultiModelService/05_MultiModelService.csproj @@ -2,21 +2,20 @@ Exe - net9.0 + net10.0 enable enable - + - diff --git a/dotnet/samples/GettingStarted/Workflows/_Foundational/05_MultiModelService/Program.cs b/dotnet/samples/GettingStarted/Workflows/_Foundational/05_MultiModelService/Program.cs index c90131a27c..7d81d891f7 100644 --- a/dotnet/samples/GettingStarted/Workflows/_Foundational/05_MultiModelService/Program.cs +++ b/dotnet/samples/GettingStarted/Workflows/_Foundational/05_MultiModelService/Program.cs @@ -12,19 +12,16 @@ IChatClient aws = new AmazonBedrockRuntimeClient( Environment.GetEnvironmentVariable("BEDROCK_ACCESSKEY"!), Environment.GetEnvironmentVariable("BEDROCK_SECRETACCESSKEY")!, - Amazon.RegionEndpoint.USEast1).AsIChatClient("amazon.nova-pro-v1:0"); + Amazon.RegionEndpoint.USEast1) + .AsIChatClient("amazon.nova-pro-v1:0"); -IChatClient anthropic = new Anthropic.SDK.AnthropicClient( - Environment.GetEnvironmentVariable("ANTHROPIC_APIKEY")!).Messages.AsBuilder() - .ConfigureOptions(o => - { - o.ModelId ??= "claude-sonnet-4-20250514"; - o.MaxOutputTokens ??= 10 * 1024; - }) - .Build(); +IChatClient anthropic = new Anthropic.AnthropicClient( + new() { APIKey = Environment.GetEnvironmentVariable("ANTHROPIC_APIKEY") }) + .AsIChatClient("claude-sonnet-4-20250514"); IChatClient openai = new OpenAI.OpenAIClient( - Environment.GetEnvironmentVariable("OPENAI_APIKEY")!).GetChatClient("gpt-4o-mini").AsIChatClient(); + Environment.GetEnvironmentVariable("OPENAI_API_KEY")!).GetChatClient("gpt-4o-mini") + .AsIChatClient(); // Define our agents. AIAgent researcher = new ChatClientAgent(aws, diff --git a/dotnet/samples/GettingStarted/Workflows/_Foundational/06_SubWorkflows/06_SubWorkflows.csproj b/dotnet/samples/GettingStarted/Workflows/_Foundational/06_SubWorkflows/06_SubWorkflows.csproj index 89b1e4bbe0..6c33744eee 100644 --- a/dotnet/samples/GettingStarted/Workflows/_Foundational/06_SubWorkflows/06_SubWorkflows.csproj +++ b/dotnet/samples/GettingStarted/Workflows/_Foundational/06_SubWorkflows/06_SubWorkflows.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable @@ -10,7 +10,6 @@ - diff --git a/dotnet/samples/GettingStarted/Workflows/_Foundational/07_MixedWorkflowAgentsAndExecutors/07_MixedWorkflowAgentsAndExecutors.csproj b/dotnet/samples/GettingStarted/Workflows/_Foundational/07_MixedWorkflowAgentsAndExecutors/07_MixedWorkflowAgentsAndExecutors.csproj index 51b18bdeb2..d0c0656ade 100644 --- a/dotnet/samples/GettingStarted/Workflows/_Foundational/07_MixedWorkflowAgentsAndExecutors/07_MixedWorkflowAgentsAndExecutors.csproj +++ b/dotnet/samples/GettingStarted/Workflows/_Foundational/07_MixedWorkflowAgentsAndExecutors/07_MixedWorkflowAgentsAndExecutors.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 enable enable @@ -16,7 +16,6 @@ - diff --git a/dotnet/samples/GettingStarted/Workflows/_Foundational/07_MixedWorkflowAgentsAndExecutors/Program.cs b/dotnet/samples/GettingStarted/Workflows/_Foundational/07_MixedWorkflowAgentsAndExecutors/Program.cs index c5437a5809..f665c1b817 100644 --- a/dotnet/samples/GettingStarted/Workflows/_Foundational/07_MixedWorkflowAgentsAndExecutors/Program.cs +++ b/dotnet/samples/GettingStarted/Workflows/_Foundational/07_MixedWorkflowAgentsAndExecutors/Program.cs @@ -132,7 +132,7 @@ private static async Task ExecuteWorkflowAsync(Workflow workflow, string input) const bool ShowAgentThinking = false; // Execute in streaming mode to see real-time progress - await using StreamingRun run = await InProcessExecution.StreamAsync(workflow, input); + await using StreamingRun run = await InProcessExecution.StreamAsync(workflow, input); // Watch the workflow events await foreach (WorkflowEvent evt in run.WatchStreamAsync()) diff --git a/dotnet/samples/GettingStarted/Workflows/_Foundational/08_WriterCriticWorkflow/08_WriterCriticWorkflow.csproj b/dotnet/samples/GettingStarted/Workflows/_Foundational/08_WriterCriticWorkflow/08_WriterCriticWorkflow.csproj index 24901257c8..d7804cef4e 100644 --- a/dotnet/samples/GettingStarted/Workflows/_Foundational/08_WriterCriticWorkflow/08_WriterCriticWorkflow.csproj +++ b/dotnet/samples/GettingStarted/Workflows/_Foundational/08_WriterCriticWorkflow/08_WriterCriticWorkflow.csproj @@ -2,7 +2,7 @@ Exe - net9.0 + net10.0 WriterCriticWorkflow enable enable @@ -11,7 +11,6 @@ - diff --git a/dotnet/samples/GettingStarted/Workflows/_Foundational/08_WriterCriticWorkflow/Program.cs b/dotnet/samples/GettingStarted/Workflows/_Foundational/08_WriterCriticWorkflow/Program.cs index fc39044b42..265a87b5f6 100644 --- a/dotnet/samples/GettingStarted/Workflows/_Foundational/08_WriterCriticWorkflow/Program.cs +++ b/dotnet/samples/GettingStarted/Workflows/_Foundational/08_WriterCriticWorkflow/Program.cs @@ -89,7 +89,7 @@ private static async Task Main() private static async Task ExecuteWorkflowAsync(Workflow workflow, string input) { // Execute in streaming mode to see real-time progress - await using StreamingRun run = await InProcessExecution.StreamAsync(workflow, input); + await using StreamingRun run = await InProcessExecution.StreamAsync(workflow, input); // Watch the workflow events await foreach (WorkflowEvent evt in run.WatchStreamAsync()) @@ -285,19 +285,19 @@ public CriticExecutor(IChatClient chatClient) : base("Critic") this._agent = new ChatClientAgent(chatClient, new ChatClientAgentOptions { Name = "Critic", - Instructions = """ - You are a constructive critic. Review the content and provide specific feedback. - Always try to provide actionable suggestions for improvement and strive to identify improvement points. - Only approve if the content is high quality, clear, and meets the original requirements and you see no improvement points. - - Provide your decision as structured output with: - - approved: true if content is good, false if revisions needed - - feedback: specific improvements needed (empty if approved) - - Be concise but specific in your feedback. - """, ChatOptions = new() { + Instructions = """ + You are a constructive critic. Review the content and provide specific feedback. + Always try to provide actionable suggestions for improvement and strive to identify improvement points. + Only approve if the content is high quality, clear, and meets the original requirements and you see no improvement points. + + Provide your decision as structured output with: + - approved: true if content is good, false if revisions needed + - feedback: specific improvements needed (empty if approved) + + Be concise but specific in your feedback. + """, ResponseFormat = ChatResponseFormat.ForJsonSchema() } }); diff --git a/dotnet/samples/HostedAgents/AgentWithHostedMCP/AgentWithHostedMCP.csproj b/dotnet/samples/HostedAgents/AgentWithHostedMCP/AgentWithHostedMCP.csproj new file mode 100644 index 0000000000..d2c0ea70f8 --- /dev/null +++ b/dotnet/samples/HostedAgents/AgentWithHostedMCP/AgentWithHostedMCP.csproj @@ -0,0 +1,70 @@ + + + + Exe + net10.0 + + enable + enable + + + false + $(NoWarn);MEAI001;OPENAI001 + + + + + + + + + + + + + + + + + + + + + + + + + all + runtime; build; native; contentfiles; analyzers; buildtransitive + + + all + runtime; build; native; contentfiles; analyzers; buildtransitive + + + all + runtime; build; native; contentfiles; analyzers; buildtransitive + + + all + runtime; build; native; contentfiles; analyzers; buildtransitive + + + all + runtime; build; native; contentfiles; analyzers; buildtransitive + + + + diff --git a/dotnet/samples/HostedAgents/AgentWithHostedMCP/Dockerfile b/dotnet/samples/HostedAgents/AgentWithHostedMCP/Dockerfile new file mode 100644 index 0000000000..a2590fc112 --- /dev/null +++ b/dotnet/samples/HostedAgents/AgentWithHostedMCP/Dockerfile @@ -0,0 +1,20 @@ +# Build the application +FROM mcr.microsoft.com/dotnet/sdk:10.0-alpine AS build +WORKDIR /src + +# Copy files from the current directory on the host to the working directory in the container +COPY . . + +RUN dotnet restore +RUN dotnet build -c Release --no-restore +RUN dotnet publish -c Release --no-build -o /app -f net10.0 + +# Run the application +FROM mcr.microsoft.com/dotnet/aspnet:10.0-alpine AS final +WORKDIR /app + +# Copy everything needed to run the app from the "build" stage. +COPY --from=build /app . + +EXPOSE 8088 +ENTRYPOINT ["dotnet", "AgentWithHostedMCP.dll"] diff --git a/dotnet/samples/HostedAgents/AgentWithHostedMCP/Program.cs b/dotnet/samples/HostedAgents/AgentWithHostedMCP/Program.cs new file mode 100644 index 0000000000..9cbea8b73a --- /dev/null +++ b/dotnet/samples/HostedAgents/AgentWithHostedMCP/Program.cs @@ -0,0 +1,34 @@ +// Copyright (c) Microsoft. All rights reserved. + +// This sample shows how to create and use a simple AI agent with OpenAI Responses as the backend, that uses a Hosted MCP Tool. +// In this case the OpenAI responses service will invoke any MCP tools as required. MCP tools are not invoked by the Agent Framework. +// The sample demonstrates how to use MCP tools with auto approval by setting ApprovalMode to NeverRequire. + +using Azure.AI.AgentServer.AgentFramework.Extensions; +using Azure.AI.OpenAI; +using Azure.Identity; +using Microsoft.Agents.AI; +using Microsoft.Extensions.AI; +using OpenAI; + +var endpoint = Environment.GetEnvironmentVariable("AZURE_OPENAI_ENDPOINT") ?? throw new InvalidOperationException("AZURE_OPENAI_ENDPOINT is not set."); +var deploymentName = Environment.GetEnvironmentVariable("AZURE_OPENAI_DEPLOYMENT_NAME") ?? "gpt-4o-mini"; + +// Create an MCP tool that can be called without approval. +AITool mcpTool = new HostedMcpServerTool(serverName: "microsoft_learn", serverAddress: "https://learn.microsoft.com/api/mcp") +{ + AllowedTools = ["microsoft_docs_search"], + ApprovalMode = HostedMcpServerToolApprovalMode.NeverRequire +}; + +// Create an agent with the MCP tool using Azure OpenAI Responses. +AIAgent agent = new AzureOpenAIClient( + new Uri(endpoint), + new DefaultAzureCredential()) + .GetOpenAIResponseClient(deploymentName) + .CreateAIAgent( + instructions: "You answer questions by searching the Microsoft Learn content only.", + name: "MicrosoftLearnAgent", + tools: [mcpTool]); + +await agent.RunAIAgentAsync(); diff --git a/dotnet/samples/HostedAgents/AgentWithHostedMCP/README.md b/dotnet/samples/HostedAgents/AgentWithHostedMCP/README.md new file mode 100644 index 0000000000..a5648d7ac9 --- /dev/null +++ b/dotnet/samples/HostedAgents/AgentWithHostedMCP/README.md @@ -0,0 +1,43 @@ +# What this sample demonstrates + +This sample demonstrates how to use a Hosted Model Context Protocol (MCP) server with an AI agent. +The agent connects to the Microsoft Learn MCP server to search documentation and answer questions using official Microsoft content. + +Key features: +- Configuring MCP tools with automatic approval (no user confirmation required) +- Filtering available tools from an MCP server +- Using Azure OpenAI Responses with MCP tools + +## Prerequisites + +Before running this sample, ensure you have: + +1. An Azure OpenAI endpoint configured +2. A deployment of a chat model (e.g., gpt-4o-mini) +3. Azure CLI installed and authenticated + +**Note**: This sample uses Azure CLI credentials for authentication. Make sure you're logged in with `az login` and have access to the Azure OpenAI resource. + +## Environment Variables + +Set the following environment variables: + +```powershell +# Replace with your Azure OpenAI endpoint +$env:AZURE_OPENAI_ENDPOINT="https://your-openai-resource.openai.azure.com/" + +# Optional, defaults to gpt-4o-mini +$env:AZURE_OPENAI_DEPLOYMENT_NAME="gpt-4o-mini" +``` + +## How It Works + +The sample connects to the Microsoft Learn MCP server and uses its documentation search capabilities: + +1. The agent is configured with a HostedMcpServerTool pointing to `https://learn.microsoft.com/api/mcp` +2. Only the `microsoft_docs_search` tool is enabled from the available MCP tools +3. Approval mode is set to `NeverRequire`, allowing automatic tool execution +4. When you ask questions, Azure OpenAI Responses automatically invokes the MCP tool to search documentation +5. The agent returns answers based on the Microsoft Learn content + +In this configuration, the OpenAI Responses service manages tool invocation directly - the Agent Framework does not handle MCP tool calls. diff --git a/dotnet/samples/HostedAgents/AgentWithHostedMCP/agent.yaml b/dotnet/samples/HostedAgents/AgentWithHostedMCP/agent.yaml new file mode 100644 index 0000000000..6444f1aad0 --- /dev/null +++ b/dotnet/samples/HostedAgents/AgentWithHostedMCP/agent.yaml @@ -0,0 +1,31 @@ +name: AgentWithHostedMCP +displayName: "Microsoft Learn Response Agent with MCP" +description: > + An AI agent that uses Azure OpenAI Responses with a Hosted Model Context Protocol (MCP) server. + The agent answers questions by searching Microsoft Learn documentation using MCP tools. + This demonstrates how MCP tools can be integrated with Azure OpenAI Responses where the service + itself handles tool invocation. +metadata: + authors: + - Microsoft Agent Framework Team + tags: + - Azure AI AgentServer + - Microsoft Agent Framework + - Model Context Protocol + - MCP + - Tool Call Approval +template: + kind: hosted + name: AgentWithHostedMCP + protocols: + - protocol: responses + version: v1 + environment_variables: + - name: AZURE_OPENAI_ENDPOINT + value: ${AZURE_OPENAI_ENDPOINT} + - name: AZURE_OPENAI_DEPLOYMENT_NAME + value: gpt-4o-mini +resources: + - name: "gpt-4o-mini" + kind: model + id: gpt-4o-mini diff --git a/dotnet/samples/HostedAgents/AgentWithHostedMCP/run-requests.http b/dotnet/samples/HostedAgents/AgentWithHostedMCP/run-requests.http new file mode 100644 index 0000000000..cc26f43b90 --- /dev/null +++ b/dotnet/samples/HostedAgents/AgentWithHostedMCP/run-requests.http @@ -0,0 +1,30 @@ +@host = http://localhost:8088 +@endpoint = {{host}}/responses + +### Health Check +GET {{host}}/readiness + +### Simple string input - Ask about MCP Tools +POST {{endpoint}} +Content-Type: application/json +{ + "input": "Please summarize the Azure AI Agent documentation related to MCP Tool calling?" +} + +### Explicit input - Ask about Agent Framework +POST {{endpoint}} +Content-Type: application/json +{ + "input": [ + { + "type": "message", + "role": "user", + "content": [ + { + "type": "input_text", + "text": "What is the Microsoft Agent Framework?" + } + ] + } + ] +} diff --git a/dotnet/samples/HostedAgents/AgentWithTextSearchRag/AgentWithTextSearchRag.csproj b/dotnet/samples/HostedAgents/AgentWithTextSearchRag/AgentWithTextSearchRag.csproj new file mode 100644 index 0000000000..e67846f54c --- /dev/null +++ b/dotnet/samples/HostedAgents/AgentWithTextSearchRag/AgentWithTextSearchRag.csproj @@ -0,0 +1,69 @@ + + + + Exe + net10.0 + + enable + enable + + + false + + + + + + + + + + + + + + + + + + + + + + + + + all + runtime; build; native; contentfiles; analyzers; buildtransitive + + + all + runtime; build; native; contentfiles; analyzers; buildtransitive + + + all + runtime; build; native; contentfiles; analyzers; buildtransitive + + + all + runtime; build; native; contentfiles; analyzers; buildtransitive + + + all + runtime; build; native; contentfiles; analyzers; buildtransitive + + + + diff --git a/dotnet/samples/HostedAgents/AgentWithTextSearchRag/Dockerfile b/dotnet/samples/HostedAgents/AgentWithTextSearchRag/Dockerfile new file mode 100644 index 0000000000..3d944c9883 --- /dev/null +++ b/dotnet/samples/HostedAgents/AgentWithTextSearchRag/Dockerfile @@ -0,0 +1,20 @@ +# Build the application +FROM mcr.microsoft.com/dotnet/sdk:10.0-alpine AS build +WORKDIR /src + +# Copy files from the current directory on the host to the working directory in the container +COPY . . + +RUN dotnet restore +RUN dotnet build -c Release --no-restore +RUN dotnet publish -c Release --no-build -o /app -f net10.0 + +# Run the application +FROM mcr.microsoft.com/dotnet/aspnet:10.0-alpine AS final +WORKDIR /app + +# Copy everything needed to run the app from the "build" stage. +COPY --from=build /app . + +EXPOSE 8088 +ENTRYPOINT ["dotnet", "AgentWithTextSearchRag.dll"] diff --git a/dotnet/samples/GettingStarted/Agents/Agent_Step18_TextSearchRag/Program.cs b/dotnet/samples/HostedAgents/AgentWithTextSearchRag/Program.cs similarity index 84% rename from dotnet/samples/GettingStarted/Agents/Agent_Step18_TextSearchRag/Program.cs rename to dotnet/samples/HostedAgents/AgentWithTextSearchRag/Program.cs index 65f3a9e98f..d8be12c5b5 100644 --- a/dotnet/samples/GettingStarted/Agents/Agent_Step18_TextSearchRag/Program.cs +++ b/dotnet/samples/HostedAgents/AgentWithTextSearchRag/Program.cs @@ -4,6 +4,7 @@ // capabilities to an AI agent. The provider runs a search against an external knowledge base // before each model invocation and injects the results into the model context. +using Azure.AI.AgentServer.AgentFramework.Extensions; using Azure.AI.OpenAI; using Azure.Identity; using Microsoft.Agents.AI; @@ -23,7 +24,7 @@ AIAgent agent = new AzureOpenAIClient( new Uri(endpoint), - new AzureCliCredential()) + new DefaultAzureCredential()) .GetChatClient(deploymentName) .CreateAIAgent(new ChatClientAgentOptions { @@ -31,22 +32,13 @@ AIContextProviderFactory = ctx => new TextSearchProvider(MockSearchAsync, ctx.SerializedState, ctx.JsonSerializerOptions, textSearchOptions) }); -AgentThread thread = agent.GetNewThread(); - -Console.WriteLine(">> Asking about returns\n"); -Console.WriteLine(await agent.RunAsync("Hi! I need help understanding the return policy.", thread)); - -Console.WriteLine("\n>> Asking about shipping\n"); -Console.WriteLine(await agent.RunAsync("How long does standard shipping usually take?", thread)); - -Console.WriteLine("\n>> Asking about product care\n"); -Console.WriteLine(await agent.RunAsync("What is the best way to maintain the TrailRunner tent fabric?", thread)); +await agent.RunAIAgentAsync(); static Task> MockSearchAsync(string query, CancellationToken cancellationToken) { // The mock search inspects the user's question and returns pre-defined snippets // that resemble documents stored in an external knowledge source. - List results = new(); + List results = []; if (query.Contains("return", StringComparison.OrdinalIgnoreCase) || query.Contains("refund", StringComparison.OrdinalIgnoreCase)) { diff --git a/dotnet/samples/Catalog/AgentWithTextSearchRag/README.md b/dotnet/samples/HostedAgents/AgentWithTextSearchRag/README.md similarity index 100% rename from dotnet/samples/Catalog/AgentWithTextSearchRag/README.md rename to dotnet/samples/HostedAgents/AgentWithTextSearchRag/README.md diff --git a/dotnet/samples/HostedAgents/AgentWithTextSearchRag/agent.yaml b/dotnet/samples/HostedAgents/AgentWithTextSearchRag/agent.yaml new file mode 100644 index 0000000000..1366071b17 --- /dev/null +++ b/dotnet/samples/HostedAgents/AgentWithTextSearchRag/agent.yaml @@ -0,0 +1,31 @@ +name: AgentWithTextSearchRag +displayName: "Text Search RAG Agent" +description: > + An AI agent that uses TextSearchProvider for retrieval augmented generation (RAG) capabilities. + The agent runs searches against an external knowledge base before each model invocation and + injects the results into the model context. It can answer questions about Contoso Outdoors + policies and products, including return policies, refunds, shipping options, and product care + instructions such as tent maintenance. +metadata: + authors: + - Microsoft Agent Framework Team + tags: + - Azure AI AgentServer + - Microsoft Agent Framework + - Retrieval-Augmented Generation + - RAG +template: + kind: hosted + name: AgentWithTextSearchRag + protocols: + - protocol: responses + version: v1 + environment_variables: + - name: AZURE_OPENAI_ENDPOINT + value: ${AZURE_OPENAI_ENDPOINT} + - name: AZURE_OPENAI_DEPLOYMENT_NAME + value: gpt-4o-mini +resources: + - name: "gpt-4o-mini" + kind: model + id: gpt-4o-mini diff --git a/dotnet/samples/HostedAgents/AgentWithTextSearchRag/run-requests.http b/dotnet/samples/HostedAgents/AgentWithTextSearchRag/run-requests.http new file mode 100644 index 0000000000..4bfb02d8f8 --- /dev/null +++ b/dotnet/samples/HostedAgents/AgentWithTextSearchRag/run-requests.http @@ -0,0 +1,30 @@ +@host = http://localhost:8088 +@endpoint = {{host}}/responses + +### Health Check +GET {{host}}/readiness + +### Simple string input +POST {{endpoint}} +Content-Type: application/json +{ + "input": "Hi! I need help understanding the return policy." +} + +### Explicit input +POST {{endpoint}} +Content-Type: application/json +{ + "input": [ + { + "type": "message", + "role": "user", + "content": [ + { + "type": "input_text", + "text": "How long does standard shipping usually take?" + } + ] + } + ] +} diff --git a/dotnet/samples/HostedAgents/AgentsInWorkflows/AgentsInWorkflows.csproj b/dotnet/samples/HostedAgents/AgentsInWorkflows/AgentsInWorkflows.csproj new file mode 100644 index 0000000000..a865f43be5 --- /dev/null +++ b/dotnet/samples/HostedAgents/AgentsInWorkflows/AgentsInWorkflows.csproj @@ -0,0 +1,69 @@ + + + + Exe + net10.0 + + enable + enable + + + false + + + + + + + + + + + + + + + + + + + + + + + + + all + runtime; build; native; contentfiles; analyzers; buildtransitive + + + all + runtime; build; native; contentfiles; analyzers; buildtransitive + + + all + runtime; build; native; contentfiles; analyzers; buildtransitive + + + all + runtime; build; native; contentfiles; analyzers; buildtransitive + + + all + runtime; build; native; contentfiles; analyzers; buildtransitive + + + + diff --git a/dotnet/samples/HostedAgents/AgentsInWorkflows/Dockerfile b/dotnet/samples/HostedAgents/AgentsInWorkflows/Dockerfile new file mode 100644 index 0000000000..86b6c156f3 --- /dev/null +++ b/dotnet/samples/HostedAgents/AgentsInWorkflows/Dockerfile @@ -0,0 +1,20 @@ +# Build the application +FROM mcr.microsoft.com/dotnet/sdk:10.0-alpine AS build +WORKDIR /src + +# Copy files from the current directory on the host to the working directory in the container +COPY . . + +RUN dotnet restore +RUN dotnet build -c Release --no-restore +RUN dotnet publish -c Release --no-build -o /app -f net10.0 + +# Run the application +FROM mcr.microsoft.com/dotnet/aspnet:10.0-alpine AS final +WORKDIR /app + +# Copy everything needed to run the app from the "build" stage. +COPY --from=build /app . + +EXPOSE 8088 +ENTRYPOINT ["dotnet", "AgentsInWorkflows.dll"] diff --git a/dotnet/samples/Catalog/AgentsInWorkflows/Program.cs b/dotnet/samples/HostedAgents/AgentsInWorkflows/Program.cs similarity index 63% rename from dotnet/samples/Catalog/AgentsInWorkflows/Program.cs rename to dotnet/samples/HostedAgents/AgentsInWorkflows/Program.cs index 3e01f6e717..b1d8a922fd 100644 --- a/dotnet/samples/Catalog/AgentsInWorkflows/Program.cs +++ b/dotnet/samples/HostedAgents/AgentsInWorkflows/Program.cs @@ -4,6 +4,7 @@ // Three translation agents are connected sequentially to create a translation chain: // English → French → Spanish → English, showing how agents can be composed as workflow executors. +using Azure.AI.AgentServer.AgentFramework.Extensions; using Azure.AI.OpenAI; using Azure.Identity; using Microsoft.Agents.AI; @@ -14,7 +15,7 @@ var endpoint = Environment.GetEnvironmentVariable("AZURE_OPENAI_ENDPOINT") ?? throw new InvalidOperationException("AZURE_OPENAI_ENDPOINT is not set."); var deploymentName = Environment.GetEnvironmentVariable("AZURE_OPENAI_DEPLOYMENT_NAME") ?? "gpt-4o-mini"; -IChatClient chatClient = new AzureOpenAIClient(new Uri(endpoint), new AzureCliCredential()) +IChatClient chatClient = new AzureOpenAIClient(new Uri(endpoint), new DefaultAzureCredential()) .GetChatClient(deploymentName) .AsIChatClient(); @@ -23,26 +24,14 @@ AIAgent spanishAgent = GetTranslationAgent("Spanish", chatClient); AIAgent englishAgent = GetTranslationAgent("English", chatClient); -// Build the workflow by adding executors and connecting them -Workflow workflow = new WorkflowBuilder(frenchAgent) +// Build the workflow and turn it into an agent +AIAgent agent = new WorkflowBuilder(frenchAgent) .AddEdge(frenchAgent, spanishAgent) .AddEdge(spanishAgent, englishAgent) -.Build(); - -// Execute the workflow -await using StreamingRun run = await InProcessExecution.StreamAsync(workflow, new ChatMessage(ChatRole.User, "Hello World!")); - -// Must send the turn token to trigger the agents. -// The agents are wrapped as executors. When they receive messages, -// they will cache the messages and only start processing when they receive a TurnToken. -await run.TrySendMessageAsync(new TurnToken(emitEvents: true)); -await foreach (WorkflowEvent evt in run.WatchStreamAsync()) -{ - if (evt is AgentRunUpdateEvent executorComplete) - { - Console.WriteLine($"{executorComplete.ExecutorId}: {executorComplete.Data}"); - } -} + .Build() + .AsAgent(); + +await agent.RunAIAgentAsync(); static ChatClientAgent GetTranslationAgent(string targetLanguage, IChatClient chatClient) => new(chatClient, $"You are a translation assistant that translates the provided text to {targetLanguage}."); diff --git a/dotnet/samples/Catalog/AgentsInWorkflows/README.md b/dotnet/samples/HostedAgents/AgentsInWorkflows/README.md similarity index 98% rename from dotnet/samples/Catalog/AgentsInWorkflows/README.md rename to dotnet/samples/HostedAgents/AgentsInWorkflows/README.md index a92012157e..5f6babc755 100644 --- a/dotnet/samples/Catalog/AgentsInWorkflows/README.md +++ b/dotnet/samples/HostedAgents/AgentsInWorkflows/README.md @@ -13,7 +13,7 @@ The agents are connected sequentially, creating a translation chain that demonst Before you begin, ensure you have the following prerequisites: -- .NET 8.0 SDK or later +- .NET 10 SDK or later - Azure OpenAI service endpoint and deployment configured - Azure CLI installed and authenticated (for Azure credential authentication) diff --git a/dotnet/samples/HostedAgents/AgentsInWorkflows/agent.yaml b/dotnet/samples/HostedAgents/AgentsInWorkflows/agent.yaml new file mode 100644 index 0000000000..900f05d513 --- /dev/null +++ b/dotnet/samples/HostedAgents/AgentsInWorkflows/agent.yaml @@ -0,0 +1,28 @@ +name: AgentsInWorkflows +displayName: "Translation Chain Workflow Agent" +description: > + A workflow agent that performs sequential translation through multiple languages. + The agent translates text from English to French, then to Spanish, and finally back + to English, leveraging AI-powered translation capabilities in a pipeline workflow. +metadata: + authors: + - Microsoft Agent Framework Team + tags: + - Azure AI AgentServer + - Microsoft Agent Framework + - Workflows +template: + kind: hosted + name: AgentsInWorkflows + protocols: + - protocol: responses + version: v1 + environment_variables: + - name: AZURE_OPENAI_ENDPOINT + value: ${AZURE_OPENAI_ENDPOINT} + - name: AZURE_OPENAI_DEPLOYMENT_NAME + value: gpt-4o-mini +resources: + - name: "gpt-4o-mini" + kind: model + id: gpt-4o-mini diff --git a/dotnet/samples/HostedAgents/AgentsInWorkflows/run-requests.http b/dotnet/samples/HostedAgents/AgentsInWorkflows/run-requests.http new file mode 100644 index 0000000000..5c33700a93 --- /dev/null +++ b/dotnet/samples/HostedAgents/AgentsInWorkflows/run-requests.http @@ -0,0 +1,30 @@ +@host = http://localhost:8088 +@endpoint = {{host}}/responses + +### Health Check +GET {{host}}/readiness + +### Simple string input +POST {{endpoint}} +Content-Type: application/json +{ + "input": "Hello, how are you today?" +} + +### Explicit input +POST {{endpoint}} +Content-Type: application/json +{ + "input": [ + { + "type": "message", + "role": "user", + "content": [ + { + "type": "input_text", + "text": "Hello, how are you today?" + } + ] + } + ] +} diff --git a/dotnet/samples/M365Agent/AFAgentApplication.cs b/dotnet/samples/M365Agent/AFAgentApplication.cs new file mode 100644 index 0000000000..04aabf96a8 --- /dev/null +++ b/dotnet/samples/M365Agent/AFAgentApplication.cs @@ -0,0 +1,188 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json; +using AdaptiveCards; +using M365Agent.Agents; +using Microsoft.Agents.AI; +using Microsoft.Agents.Builder; +using Microsoft.Agents.Builder.App; +using Microsoft.Agents.Builder.State; +using Microsoft.Agents.Core.Models; +using Microsoft.Extensions.AI; +using Microsoft.Extensions.DependencyInjection; + +namespace M365Agent; + +/// +/// An adapter class that exposes a Microsoft Agent Framework as a M365 Agent SDK . +/// +internal sealed class AFAgentApplication : AgentApplication +{ + private readonly AIAgent _agent; + private readonly string? _welcomeMessage; + + public AFAgentApplication(AIAgent agent, AgentApplicationOptions options, [FromKeyedServices("AFAgentApplicationWelcomeMessage")] string? welcomeMessage = null) : base(options) + { + this._agent = agent; + this._welcomeMessage = welcomeMessage; + + this.OnConversationUpdate(ConversationUpdateEvents.MembersAdded, this.WelcomeMessageAsync); + this.OnActivity(ActivityTypes.Message, this.MessageActivityAsync, rank: RouteRank.Last); + } + + /// + /// The main agent invocation method, where each user message triggers a call to the underlying . + /// + private async Task MessageActivityAsync(ITurnContext turnContext, ITurnState turnState, CancellationToken cancellationToken) + { + // Start a Streaming Process + await turnContext.StreamingResponse.QueueInformativeUpdateAsync("Working on a response for you", cancellationToken); + + // Get the conversation history from turn state. + JsonElement threadElementStart = turnState.GetValue("conversation.chatHistory"); + + // Deserialize the conversation history into an AgentThread, or create a new one if none exists. + AgentThread agentThread = threadElementStart.ValueKind is not JsonValueKind.Undefined and not JsonValueKind.Null + ? this._agent.DeserializeThread(threadElementStart, JsonUtilities.DefaultOptions) + : this._agent.GetNewThread(); + + ChatMessage chatMessage = HandleUserInput(turnContext); + + // Invoke the WeatherForecastAgent to process the message + AgentRunResponse agentRunResponse = await this._agent.RunAsync(chatMessage, agentThread, cancellationToken: cancellationToken); + + // Check for any user input requests in the response + // and turn them into adaptive cards in the streaming response. + List? attachments = null; + HandleUserInputRequests(agentRunResponse, ref attachments); + + // Check for Adaptive Card content in the response messages + // and return them appropriately in the response. + var adaptiveCards = agentRunResponse.Messages.SelectMany(x => x.Contents).OfType().ToList(); + if (adaptiveCards.Count > 0) + { + attachments ??= []; + attachments.Add(new Attachment() + { + ContentType = "application/vnd.microsoft.card.adaptive", + Content = adaptiveCards.First().AdaptiveCardJson, + }); + } + else + { + turnContext.StreamingResponse.QueueTextChunk(agentRunResponse.Text); + } + + // If created any adaptive cards, add them to the final message. + if (attachments is not null) + { + turnContext.StreamingResponse.FinalMessage = MessageFactory.Attachment(attachments); + } + + // Serialize and save the updated conversation history back to turn state. + JsonElement threadElementEnd = agentThread.Serialize(JsonUtilities.DefaultOptions); + turnState.SetValue("conversation.chatHistory", threadElementEnd); + + // End the streaming response + await turnContext.StreamingResponse.EndStreamAsync(cancellationToken); + } + + /// + /// A method to show a welcome message when a new user joins the conversation. + /// + private async Task WelcomeMessageAsync(ITurnContext turnContext, ITurnState turnState, CancellationToken cancellationToken) + { + if (string.IsNullOrWhiteSpace(this._welcomeMessage)) + { + return; + } + + foreach (ChannelAccount member in turnContext.Activity.MembersAdded) + { + if (member.Id != turnContext.Activity.Recipient.Id) + { + await turnContext.SendActivityAsync(MessageFactory.Text(this._welcomeMessage), cancellationToken); + } + } + } + + /// + /// When a user responds to a function approval request by clicking on a card, this method converts the response + /// into the appropriate approval or rejection . + /// + /// The for the current turn. + /// The to pass to the . + private static ChatMessage HandleUserInput(ITurnContext turnContext) + { + // Check if this contains the function approval Adaptive Card response. + if (turnContext.Activity.Value is JsonElement valueElement + && valueElement.GetProperty("type").GetString() == "functionApproval" + && valueElement.GetProperty("approved") is JsonElement approvedJsonElement + && approvedJsonElement.ValueKind is JsonValueKind.True or JsonValueKind.False + && valueElement.GetProperty("requestJson") is JsonElement requestJsonElement + && requestJsonElement.ValueKind == JsonValueKind.String) + { + var requestContent = JsonSerializer.Deserialize(requestJsonElement.GetString()!, JsonUtilities.DefaultOptions); + + return new ChatMessage(ChatRole.User, [requestContent!.CreateResponse(approvedJsonElement.ValueKind == JsonValueKind.True)]); + } + + return new ChatMessage(ChatRole.User, turnContext.Activity.Text); + } + + /// + /// When the agent returns any user input requests, this method converts them into adaptive cards that + /// asks the user to approve or deny the requests. + /// + /// The that may contain the user input requests. + /// The list of to which the adaptive cards will be added. + private static void HandleUserInputRequests(AgentRunResponse response, ref List? attachments) + { + var userInputRequests = response.UserInputRequests.ToList(); + if (userInputRequests.Count > 0) + { + foreach (var functionApprovalRequest in userInputRequests.OfType()) + { + var functionApprovalRequestJson = JsonSerializer.Serialize(functionApprovalRequest, JsonUtilities.DefaultOptions); + + var card = new AdaptiveCard("1.5"); + card.Body.Add(new AdaptiveTextBlock + { + Text = "Function Call Approval Required", + Size = AdaptiveTextSize.Large, + Weight = AdaptiveTextWeight.Bolder, + HorizontalAlignment = AdaptiveHorizontalAlignment.Center + }); + card.Body.Add(new AdaptiveTextBlock + { + Text = $"Function: {functionApprovalRequest.FunctionCall.Name}" + }); + card.Body.Add(new AdaptiveActionSet() + { + Actions = + [ + new AdaptiveSubmitAction + { + Id = "Approve", + Title = "Approve", + Data = new { type = "functionApproval", approved = true, requestJson = functionApprovalRequestJson } + }, + new AdaptiveSubmitAction + { + Id = "Deny", + Title = "Deny", + Data = new { type = "functionApproval", approved = false, requestJson = functionApprovalRequestJson } + } + ] + }); + + attachments ??= []; + attachments.Add(new Attachment() + { + ContentType = "application/vnd.microsoft.card.adaptive", + Content = card.ToJson(), + }); + } + } + } +} diff --git a/dotnet/samples/M365Agent/Agents/AdaptiveCardAIContent.cs b/dotnet/samples/M365Agent/Agents/AdaptiveCardAIContent.cs new file mode 100644 index 0000000000..9b1ebee662 --- /dev/null +++ b/dotnet/samples/M365Agent/Agents/AdaptiveCardAIContent.cs @@ -0,0 +1,36 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; +using AdaptiveCards; +using Microsoft.Agents.AI; +using Microsoft.Extensions.AI; + +namespace M365Agent.Agents; + +/// +/// An type allows an to return adaptive cards as part of its response messages. +/// +internal sealed class AdaptiveCardAIContent : AIContent +{ + public AdaptiveCardAIContent(AdaptiveCard adaptiveCard) + { + this.AdaptiveCard = adaptiveCard ?? throw new ArgumentNullException(nameof(adaptiveCard)); + } + +#pragma warning disable CS8618 // Non-nullable field must contain a non-null value when exiting constructor. Consider adding the 'required' modifier or declaring as nullable. + [JsonConstructor] + public AdaptiveCardAIContent(string adaptiveCardJson) + { + this.AdaptiveCardJson = adaptiveCardJson; + } +#pragma warning restore CS8618 // Non-nullable field must contain a non-null value when exiting constructor. Consider adding the 'required' modifier or declaring as nullable. + + [JsonIgnore] + public AdaptiveCard AdaptiveCard { get; private set; } + + public string AdaptiveCardJson + { + get => this.AdaptiveCard.ToJson(); + set => this.AdaptiveCard = AdaptiveCard.FromJson(value).Card; + } +} diff --git a/dotnet/samples/M365Agent/Agents/WeatherForecastAgent.cs b/dotnet/samples/M365Agent/Agents/WeatherForecastAgent.cs new file mode 100644 index 0000000000..740b959a7a --- /dev/null +++ b/dotnet/samples/M365Agent/Agents/WeatherForecastAgent.cs @@ -0,0 +1,115 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.ComponentModel; +using System.Text.Json; +using AdaptiveCards; +using Microsoft.Agents.AI; +using Microsoft.Extensions.AI; + +namespace M365Agent.Agents; + +/// +/// A weather forecasting agent. This agent wraps a and adds custom logic +/// to generate adaptive cards for weather forecasts and add these to the agent's response. +/// +public class WeatherForecastAgent : DelegatingAIAgent +{ + private const string AgentName = "WeatherForecastAgent"; + private const string AgentInstructions = """ + You are a friendly assistant that helps people find a weather forecast for a given location. + You may ask follow up questions until you have enough information to answer the customers question. + When answering with a weather forecast, fill out the weatherCard property with an adaptive card containing the weather information and + add some emojis to indicate the type of weather. + When answering with just text, fill out the context property with a friendly response. + """; + + /// + /// Initializes a new instance of the class. + /// + /// An instance of for interacting with an LLM. + public WeatherForecastAgent(IChatClient chatClient) + : base(new ChatClientAgent( + chatClient: chatClient, + new ChatClientAgentOptions() + { + Name = AgentName, + ChatOptions = new ChatOptions() + { + Instructions = AgentInstructions, + Tools = [new ApprovalRequiredAIFunction(AIFunctionFactory.Create(GetWeather))], + // We want the agent to return structured output in a known format + // so that we can easily create adaptive cards from the response. + ResponseFormat = ChatResponseFormat.ForJsonSchema( + schema: AIJsonUtilities.CreateJsonSchema(typeof(WeatherForecastAgentResponse)), + schemaName: "WeatherForecastAgentResponse", + schemaDescription: "Response to a query about the weather in a specified location"), + } + })) + { + } + + public override async Task RunAsync(IEnumerable messages, AgentThread? thread = null, AgentRunOptions? options = null, CancellationToken cancellationToken = default) + { + var response = await base.RunAsync(messages, thread, options, cancellationToken); + + // If the agent returned a valid structured output response + // we might be able to enhance the response with an adaptive card. + if (response.TryDeserialize(JsonSerializerOptions.Web, out var structuredOutput)) + { + var textContentMessage = response.Messages.FirstOrDefault(x => x.Contents.OfType().Any()); + if (textContentMessage is not null) + { + // If the response contains weather information, create an adaptive card. + if (structuredOutput.ContentType == WeatherForecastAgentResponseContentType.WeatherForecastAgentResponse) + { + var card = CreateWeatherCard(structuredOutput.Location, structuredOutput.MeteorologicalCondition, structuredOutput.TemperatureInCelsius); + textContentMessage.Contents.Add(new AdaptiveCardAIContent(card)); + } + + // If the response is just text, replace the structured output with the text response. + if (structuredOutput.ContentType == WeatherForecastAgentResponseContentType.OtherAgentResponse) + { + var textContent = textContentMessage.Contents.OfType().First(); + textContent.Text = structuredOutput.OtherResponse; + } + } + } + + return response; + } + + /// + /// A mock weather tool, to get weather information for a given location. + /// + [Description("Get the weather for a given location.")] + private static string GetWeather([Description("The location to get the weather for.")] string location) + => $"The weather in {location} is cloudy with a high of 15°C."; + + /// + /// Create an adaptive card to display weather information. + /// + private static AdaptiveCard CreateWeatherCard(string? location, string? condition, string? temperature) + { + var card = new AdaptiveCard("1.5"); + card.Body.Add(new AdaptiveTextBlock + { + Text = "🌤️ Weather Forecast 🌤️", + Size = AdaptiveTextSize.Large, + Weight = AdaptiveTextWeight.Bolder, + HorizontalAlignment = AdaptiveHorizontalAlignment.Center + }); + card.Body.Add(new AdaptiveTextBlock + { + Text = "Location: " + location, + }); + card.Body.Add(new AdaptiveTextBlock + { + Text = "Condition: " + condition, + }); + card.Body.Add(new AdaptiveTextBlock + { + Text = "Temperature: " + temperature, + }); + return card; + } +} diff --git a/dotnet/samples/M365Agent/Agents/WeatherForecastAgentResponse.cs b/dotnet/samples/M365Agent/Agents/WeatherForecastAgentResponse.cs new file mode 100644 index 0000000000..e5e15dffd4 --- /dev/null +++ b/dotnet/samples/M365Agent/Agents/WeatherForecastAgentResponse.cs @@ -0,0 +1,47 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.ComponentModel; +using System.Text.Json.Serialization; + +namespace M365Agent.Agents; + +/// +/// The structured output type for the . +/// +internal sealed class WeatherForecastAgentResponse +{ + /// + /// A value indicating whether the response contains a weather forecast or some other type of response. + /// + [JsonPropertyName("contentType")] + [JsonConverter(typeof(JsonStringEnumConverter))] + public WeatherForecastAgentResponseContentType ContentType { get; set; } + + /// + /// If the agent could not provide a weather forecast this should contain a textual response. + /// + [Description("If the answer is other agent response, contains the textual agent response.")] + [JsonPropertyName("otherResponse")] + public string? OtherResponse { get; set; } + + /// + /// The location for which the weather forecast is given. + /// + [Description("If the answer is a weather forecast, contains the location for which the forecast is given.")] + [JsonPropertyName("location")] + public string? Location { get; set; } + + /// + /// The temperature in Celsius for the given location. + /// + [Description("If the answer is a weather forecast, contains the temperature in Celsius.")] + [JsonPropertyName("temperatureInCelsius")] + public string? TemperatureInCelsius { get; set; } + + /// + /// The meteorological condition for the given location. + /// + [Description("If the answer is a weather forecast, contains the meteorological condition (e.g., Sunny, Rainy).")] + [JsonPropertyName("meteorologicalCondition")] + public string? MeteorologicalCondition { get; set; } +} diff --git a/dotnet/samples/M365Agent/Agents/WeatherForecastAgentResponseContentType.cs b/dotnet/samples/M365Agent/Agents/WeatherForecastAgentResponseContentType.cs new file mode 100644 index 0000000000..cd888d0a0c --- /dev/null +++ b/dotnet/samples/M365Agent/Agents/WeatherForecastAgentResponseContentType.cs @@ -0,0 +1,17 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +namespace M365Agent.Agents; + +/// +/// The type of content contained in a . +/// +internal enum WeatherForecastAgentResponseContentType +{ + [JsonPropertyName("otherAgentResponse")] + OtherAgentResponse, + + [JsonPropertyName("weatherForecastAgentResponse")] + WeatherForecastAgentResponse +} diff --git a/dotnet/samples/M365Agent/Auth/AspNetExtensions.cs b/dotnet/samples/M365Agent/Auth/AspNetExtensions.cs new file mode 100644 index 0000000000..1452c5f05e --- /dev/null +++ b/dotnet/samples/M365Agent/Auth/AspNetExtensions.cs @@ -0,0 +1,206 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Collections.Concurrent; +using System.Globalization; +using System.IdentityModel.Tokens.Jwt; +using System.Text; +using Microsoft.Agents.Authentication; +using Microsoft.Agents.Core; +using Microsoft.AspNetCore.Authentication.JwtBearer; +using Microsoft.Extensions.Configuration; +using Microsoft.Extensions.DependencyInjection; +using Microsoft.IdentityModel.Protocols; +using Microsoft.IdentityModel.Protocols.OpenIdConnect; +using Microsoft.IdentityModel.Tokens; +using Microsoft.IdentityModel.Validators; + +namespace M365Agent; + +internal static class AspNetExtensions +{ + private static readonly CompositeFormat s_cachedValidTokenIssuerUrlTemplateV1Format = CompositeFormat.Parse(AuthenticationConstants.ValidTokenIssuerUrlTemplateV1); + private static readonly CompositeFormat s_cachedValidTokenIssuerUrlTemplateV2Format = CompositeFormat.Parse(AuthenticationConstants.ValidTokenIssuerUrlTemplateV2); + + private static readonly ConcurrentDictionary> s_openIdMetadataCache = new(); + + /// + /// Adds AspNet token validation typical for ABS/SMBA and agent-to-agent using settings in configuration. + /// + /// The service collection to resolve dependencies. + /// Used to read configuration settings. + /// Name of the config section to read. + /// + /// This extension reads settings from configuration. If configuration is missing JWT token + /// is not enabled. + ///

The minimum, but typical, configuration is:

+ /// + /// "TokenValidation": { + /// "Enabled": boolean, + /// "Audiences": [ + /// "{{ClientId}}" // this is the Client ID used for the Azure Bot + /// ], + /// "TenantId": "{{TenantId}}" + /// } + /// + /// The full options are: + /// + /// "TokenValidation": { + /// "Enabled": boolean, + /// "Audiences": [ + /// "{required:agent-appid}" + /// ], + /// "TenantId": "{recommended:tenant-id}", + /// "ValidIssuers": [ + /// "{default:Public-AzureBotService}" + /// ], + /// "IsGov": {optional:false}, + /// "AzureBotServiceOpenIdMetadataUrl": optional, + /// "OpenIdMetadataUrl": optional, + /// "AzureBotServiceTokenHandling": "{optional:true}" + /// "OpenIdMetadataRefresh": "optional-12:00:00" + /// } + /// + ///
+ public static void AddAgentAspNetAuthentication(this IServiceCollection services, IConfiguration configuration, string tokenValidationSectionName = "TokenValidation") + { + IConfigurationSection tokenValidationSection = configuration.GetSection(tokenValidationSectionName); + + if (!tokenValidationSection.Exists() || !tokenValidationSection.GetValue("Enabled", true)) + { + // Noop if TokenValidation section missing or disabled. + System.Diagnostics.Trace.WriteLine("AddAgentAspNetAuthentication: Auth disabled"); + return; + } + + services.AddAgentAspNetAuthentication(tokenValidationSection.Get()!); + } + + /// + /// Adds AspNet token validation typical for ABS/SMBA and agent-to-agent. + /// + public static void AddAgentAspNetAuthentication(this IServiceCollection services, TokenValidationOptions validationOptions) + { + AssertionHelpers.ThrowIfNull(validationOptions, nameof(validationOptions)); + + // Must have at least one Audience. + if (validationOptions.Audiences == null || validationOptions.Audiences.Count == 0) + { + throw new ArgumentException($"{nameof(TokenValidationOptions)}:Audiences requires at least one ClientId"); + } + + // Audience values must be GUID's + foreach (var audience in validationOptions.Audiences) + { + if (!Guid.TryParse(audience, out _)) + { + throw new ArgumentException($"{nameof(TokenValidationOptions)}:Audiences values must be a GUID"); + } + } + + // If ValidIssuers is empty, default for ABS Public Cloud + if (validationOptions.ValidIssuers == null || validationOptions.ValidIssuers.Count == 0) + { + validationOptions.ValidIssuers = + [ + "https://api.botframework.com", + "https://sts.windows.net/d6d49420-f39b-4df7-a1dc-d59a935871db/", + "https://login.microsoftonline.com/d6d49420-f39b-4df7-a1dc-d59a935871db/v2.0", + "https://sts.windows.net/f8cdef31-a31e-4b4a-93e4-5f571e91255a/", + "https://login.microsoftonline.com/f8cdef31-a31e-4b4a-93e4-5f571e91255a/v2.0", + "https://sts.windows.net/69e9b82d-4842-4902-8d1e-abc5b98a55e8/", + "https://login.microsoftonline.com/69e9b82d-4842-4902-8d1e-abc5b98a55e8/v2.0", + ]; + + if (!string.IsNullOrEmpty(validationOptions.TenantId) && Guid.TryParse(validationOptions.TenantId, out _)) + { + validationOptions.ValidIssuers.Add(string.Format(CultureInfo.InvariantCulture, s_cachedValidTokenIssuerUrlTemplateV1Format, validationOptions.TenantId)); + validationOptions.ValidIssuers.Add(string.Format(CultureInfo.InvariantCulture, s_cachedValidTokenIssuerUrlTemplateV2Format, validationOptions.TenantId)); + } + } + + // If the `AzureBotServiceOpenIdMetadataUrl` setting is not specified, use the default based on `IsGov`. This is what is used to authenticate ABS tokens. + if (string.IsNullOrEmpty(validationOptions.AzureBotServiceOpenIdMetadataUrl)) + { + validationOptions.AzureBotServiceOpenIdMetadataUrl = validationOptions.IsGov ? AuthenticationConstants.GovAzureBotServiceOpenIdMetadataUrl : AuthenticationConstants.PublicAzureBotServiceOpenIdMetadataUrl; + } + + // If the `OpenIdMetadataUrl` setting is not specified, use the default based on `IsGov`. This is what is used to authenticate Entra ID tokens. + if (string.IsNullOrEmpty(validationOptions.OpenIdMetadataUrl)) + { + validationOptions.OpenIdMetadataUrl = validationOptions.IsGov ? AuthenticationConstants.GovOpenIdMetadataUrl : AuthenticationConstants.PublicOpenIdMetadataUrl; + } + + var openIdMetadataRefresh = validationOptions.OpenIdMetadataRefresh ?? BaseConfigurationManager.DefaultAutomaticRefreshInterval; + + _ = services.AddAuthentication(options => + { + options.DefaultAuthenticateScheme = JwtBearerDefaults.AuthenticationScheme; + options.DefaultChallengeScheme = JwtBearerDefaults.AuthenticationScheme; + }) + .AddJwtBearer(options => + { + options.SaveToken = true; + options.TokenValidationParameters = new TokenValidationParameters + { + ValidateIssuer = true, + ValidateAudience = true, + ValidateLifetime = true, + ClockSkew = TimeSpan.FromMinutes(5), + ValidIssuers = validationOptions.ValidIssuers, + ValidAudiences = validationOptions.Audiences, + ValidateIssuerSigningKey = true, + RequireSignedTokens = true, + }; + + // Using Microsoft.IdentityModel.Validators + options.TokenValidationParameters.EnableAadSigningKeyIssuerValidation(); + + options.Events = new JwtBearerEvents + { + // Create a ConfigurationManager based on the requestor. This is to handle ABS non-Entra tokens. + OnMessageReceived = async context => + { + string authorizationHeader = context.Request.Headers.Authorization.ToString(); + + if (string.IsNullOrWhiteSpace(authorizationHeader)) + { + // Default to AadTokenValidation handling + context.Options.TokenValidationParameters.ConfigurationManager ??= options.ConfigurationManager as BaseConfigurationManager; + await Task.CompletedTask.ConfigureAwait(false); + return; + } + + string[] parts = authorizationHeader.Split(' ')!; + if (parts.Length != 2 || parts[0] != "Bearer") + { + // Default to AadTokenValidation handling + context.Options.TokenValidationParameters.ConfigurationManager ??= options.ConfigurationManager as BaseConfigurationManager; + await Task.CompletedTask.ConfigureAwait(false); + return; + } + + JwtSecurityToken token = new(parts[1]); + string issuer = token.Claims.FirstOrDefault(claim => claim.Type == AuthenticationConstants.IssuerClaim)?.Value!; + + string openIdMetadataUrl = (validationOptions.AzureBotServiceTokenHandling && AuthenticationConstants.BotFrameworkTokenIssuer.Equals(issuer, StringComparison.Ordinal)) + ? validationOptions.AzureBotServiceOpenIdMetadataUrl + : validationOptions.OpenIdMetadataUrl; + + context.Options.TokenValidationParameters.ConfigurationManager = s_openIdMetadataCache.GetOrAdd(openIdMetadataUrl, key => + { + return new ConfigurationManager(openIdMetadataUrl, new OpenIdConnectConfigurationRetriever(), new HttpClient()) + { + AutomaticRefreshInterval = openIdMetadataRefresh + }; + }); + + await Task.CompletedTask.ConfigureAwait(false); + }, + + OnTokenValidated = context => Task.CompletedTask, + OnForbidden = context => Task.CompletedTask, + OnAuthenticationFailed = context => Task.CompletedTask + }; + }); + } +} diff --git a/dotnet/samples/M365Agent/Auth/TokenValidationOptions.cs b/dotnet/samples/M365Agent/Auth/TokenValidationOptions.cs new file mode 100644 index 0000000000..f8f2fa2e08 --- /dev/null +++ b/dotnet/samples/M365Agent/Auth/TokenValidationOptions.cs @@ -0,0 +1,52 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Microsoft.Agents.Authentication; + +namespace M365Agent; + +internal sealed class TokenValidationOptions +{ + /// + /// The list of audiences to validate against. + /// + public IList? Audiences { get; set; } + + /// + /// TenantId of the Azure Bot. Optional but recommended. + /// + public string? TenantId { get; set; } + + /// + /// Additional valid issuers. Optional, in which case the Public Azure Bot Service issuers are used. + /// + public IList? ValidIssuers { get; set; } + + /// + /// Can be omitted, in which case public Azure Bot Service and Azure Cloud metadata urls are used. + /// + public bool IsGov { get; set; } + + /// + /// Azure Bot Service OpenIdMetadataUrl. Optional, in which case default value depends on IsGov. + /// + /// + /// + public string? AzureBotServiceOpenIdMetadataUrl { get; set; } + + /// + /// Entra OpenIdMetadataUrl. Optional, in which case default value depends on IsGov. + /// + /// + /// + public string? OpenIdMetadataUrl { get; set; } + + /// + /// Determines if Azure Bot Service tokens are handled. Defaults to true and should always be true until Azure Bot Service sends Entra ID token. + /// + public bool AzureBotServiceTokenHandling { get; set; } = true; + + /// + /// OpenIdMetadata refresh interval. Defaults to 12 hours. + /// + public TimeSpan? OpenIdMetadataRefresh { get; set; } +} diff --git a/dotnet/samples/M365Agent/JsonUtilities.cs b/dotnet/samples/M365Agent/JsonUtilities.cs new file mode 100644 index 0000000000..c87367e65b --- /dev/null +++ b/dotnet/samples/M365Agent/JsonUtilities.cs @@ -0,0 +1,73 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Diagnostics.CodeAnalysis; +using System.Text.Encodings.Web; +using System.Text.Json; +using System.Text.Json.Serialization; +using System.Text.Json.Serialization.Metadata; +using M365Agent.Agents; +using Microsoft.Extensions.AI; + +namespace M365Agent; + +/// Provides a collection of utility methods for working with JSON data in the context of the application. +internal static partial class JsonUtilities +{ + /// + /// Gets the singleton used as the default in JSON serialization operations. + /// + /// + /// + /// For Native AOT or applications disabling , this instance + /// includes source generated contracts for all common exchange types contained in this library. + /// + /// + /// It additionally turns on the following settings: + /// + /// Enables defaults. + /// Enables as the default ignore condition for properties. + /// Enables as the default number handling for number types. + /// + /// + /// + public static JsonSerializerOptions DefaultOptions { get; } = CreateDefaultOptions(); + + /// + /// Creates default options to use for agents-related serialization. + /// + /// The configured options. + [UnconditionalSuppressMessage("ReflectionAnalysis", "IL3050:RequiresDynamicCode", Justification = "Converter is guarded by IsReflectionEnabledByDefault check.")] + [UnconditionalSuppressMessage("Trimming", "IL2026:Members annotated with 'RequiresUnreferencedCodeAttribute' require dynamic access", Justification = "Converter is guarded by IsReflectionEnabledByDefault check.")] + private static JsonSerializerOptions CreateDefaultOptions() + { + // Copy the configuration from the source generated context. + JsonSerializerOptions options = new(JsonContext.Default.Options) + { + // Chain in the resolvers from both AgentAbstractionsJsonUtilities and our source generated context. + // We want AgentAbstractionsJsonUtilities first to ensure any M.E.AI types are handled via its resolver. + TypeInfoResolver = JsonTypeInfoResolver.Combine(AIJsonUtilities.DefaultOptions.TypeInfoResolver, JsonContext.Default), + Encoder = JavaScriptEncoder.UnsafeRelaxedJsonEscaping, // same as in AgentAbstractionsJsonUtilities and AIJsonUtilities + }; + options.AddAIContentType(typeDiscriminatorId: "adaptiveCard"); + + if (JsonSerializer.IsReflectionEnabledByDefault) + { + options.Converters.Add(new JsonStringEnumConverter()); + } + + options.MakeReadOnly(); + return options; + } + + // Keep in sync with CreateDefaultOptions above. + [JsonSourceGenerationOptions(JsonSerializerDefaults.Web, + UseStringEnumConverter = true, + DefaultIgnoreCondition = JsonIgnoreCondition.WhenWritingNull, + NumberHandling = JsonNumberHandling.AllowReadingFromString)] + + // M365Agent specific types + [JsonSerializable(typeof(AdaptiveCardAIContent))] + + [ExcludeFromCodeCoverage] + internal sealed partial class JsonContext : JsonSerializerContext; +} diff --git a/dotnet/samples/M365Agent/M365Agent.csproj b/dotnet/samples/M365Agent/M365Agent.csproj new file mode 100644 index 0000000000..f40d404204 --- /dev/null +++ b/dotnet/samples/M365Agent/M365Agent.csproj @@ -0,0 +1,29 @@ + + + + Exe + net10.0 + enable + enable + b842df34-390f-490d-9dc0-73909363ad16 + $(NoWarn);CA1812 + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/dotnet/samples/M365Agent/Program.cs b/dotnet/samples/M365Agent/Program.cs new file mode 100644 index 0000000000..834ac2180a --- /dev/null +++ b/dotnet/samples/M365Agent/Program.cs @@ -0,0 +1,104 @@ +// Copyright (c) Microsoft. All rights reserved. + +// Sample that shows how to create an Agent Framework agent that is hosted using the M365 Agent SDK. +// The agent can then be consumed from various M365 channels. +// See the README.md for more information. + +using Azure.AI.OpenAI; +using Azure.Identity; +using M365Agent; +using M365Agent.Agents; +using Microsoft.Agents.AI; +using Microsoft.Agents.Builder; +using Microsoft.Agents.Hosting.AspNetCore; +using Microsoft.Agents.Storage; +using Microsoft.AspNetCore.Builder; +using Microsoft.AspNetCore.Http; +using Microsoft.Extensions.AI; +using Microsoft.Extensions.Configuration; +using Microsoft.Extensions.DependencyInjection; +using Microsoft.Extensions.Hosting; +using OpenAI; + +WebApplicationBuilder builder = WebApplication.CreateBuilder(args); + +if (builder.Environment.IsDevelopment()) +{ + builder.Configuration.AddUserSecrets(); +} + +builder.Services.AddHttpClient(); + +// Register the inference service of your choice. AzureOpenAI and OpenAI are demonstrated... +IChatClient chatClient; +if (builder.Configuration.GetSection("AIServices").GetValue("UseAzureOpenAI")) +{ + var deploymentName = builder.Configuration.GetSection("AIServices:AzureOpenAI").GetValue("DeploymentName")!; + var endpoint = builder.Configuration.GetSection("AIServices:AzureOpenAI").GetValue("Endpoint")!; + + chatClient = new AzureOpenAIClient( + new Uri(endpoint), + new AzureCliCredential()) + .GetChatClient(deploymentName) + .AsIChatClient(); +} +else +{ + var modelId = builder.Configuration.GetSection("AIServices:OpenAI").GetValue("ModelId")!; + var apiKey = builder.Configuration.GetSection("AIServices:OpenAI").GetValue("ApiKey")!; + + chatClient = new OpenAIClient( + apiKey) + .GetChatClient(modelId) + .AsIChatClient(); +} +builder.Services.AddSingleton(chatClient); + +// Add AgentApplicationOptions from appsettings section "AgentApplication". +builder.AddAgentApplicationOptions(); + +// Add the WeatherForecastAgent plus a welcome message. +// These will be consumed by the AFAgentApplication and exposed as an Agent SDK AgentApplication. +builder.Services.AddSingleton(); +builder.Services.AddKeyedSingleton("AFAgentApplicationWelcomeMessage", "Hello and Welcome! I'm here to help with all your weather forecast needs!"); + +// Add the AgentApplication, which contains the logic for responding to +// user messages via the Agent SDK. +builder.AddAgent(); + +// Register IStorage. For development, MemoryStorage is suitable. +// For production Agents, persisted storage should be used so +// that state survives Agent restarts, and operates correctly +// in a cluster of Agent instances. +builder.Services.AddSingleton(); + +// Configure the HTTP request pipeline. + +// Add AspNet token validation for Azure Bot Service and Entra. Authentication is +// configured in the appsettings.json "TokenValidation" section. +builder.Services.AddControllers(); +builder.Services.AddAgentAspNetAuthentication(builder.Configuration); + +WebApplication app = builder.Build(); + +// Enable AspNet authentication and authorization +app.UseAuthentication(); +app.UseAuthorization(); + +app.MapGet("/", () => "Microsoft Agents SDK Sample"); + +// This receives incoming messages and routes them to the registered AgentApplication. +var incomingRoute = app.MapPost("/api/messages", async (HttpRequest request, HttpResponse response, IAgentHttpAdapter adapter, IAgent agent, CancellationToken cancellationToken) => await adapter.ProcessAsync(request, response, agent, cancellationToken)); + +if (!app.Environment.IsDevelopment()) +{ + incomingRoute.RequireAuthorization(); +} +else +{ + // Hardcoded for brevity and ease of testing. + // In production, this should be set in configuration. + app.Urls.Add("http://localhost:3978"); +} + +app.Run(); diff --git a/dotnet/samples/M365Agent/Properties/launchSettings.json b/dotnet/samples/M365Agent/Properties/launchSettings.json new file mode 100644 index 0000000000..14d89c0bab --- /dev/null +++ b/dotnet/samples/M365Agent/Properties/launchSettings.json @@ -0,0 +1,12 @@ +{ + "profiles": { + "M365Agent": { + "commandName": "Project", + "launchBrowser": true, + "environmentVariables": { + "ASPNETCORE_ENVIRONMENT": "Development" + }, + "applicationUrl": "https://localhost:49692;http://localhost:49693" + } + } +} \ No newline at end of file diff --git a/dotnet/samples/M365Agent/README.md b/dotnet/samples/M365Agent/README.md new file mode 100644 index 0000000000..e61fa438f5 --- /dev/null +++ b/dotnet/samples/M365Agent/README.md @@ -0,0 +1,119 @@ +# Microsoft Agent Framework agents with the M365 Agents SDK Weather Agent sample + +This is a sample of a simple Weather Forecast Agent that is hosted on an Asp.Net core web service and is exposed via the M365 Agent SDK. This Agent is configured to accept a request asking for information about a weather forecast and respond to the caller with an Adaptive Card. This agent will handle multiple "turns" to get the required information from the user. + +This Agent Sample is intended to introduce you the basics of integrating Agent Framework with the Microsoft 365 Agents SDK in order to use Agent Framework agents in various M365 services and applications. It can also be used as the base for a custom Agent that you choose to develop. + +***Note:*** This sample requires JSON structured output from the model which works best from newer versions of the model such as gpt-4o-mini. + +## Prerequisites + +- [.NET 10.0 SDK or later](https://dotnet.microsoft.com/download) +- [devtunnel](https://learn.microsoft.com/azure/developer/dev-tunnels/get-started?tabs=windows) +- [Microsoft 365 Agents Toolkit](https://github.com/OfficeDev/microsoft-365-agents-toolkit) + +- You will need an Azure OpenAI or OpenAI resource using `gpt-4o-mini` + +- Configure OpenAI in appsettings + + ```json + "AIServices": { + "AzureOpenAI": { + "DeploymentName": "", // This is the Deployment (as opposed to model) Name of the Azure OpenAI model + "Endpoint": "", // This is the Endpoint of the Azure OpenAI resource + "ApiKey": "" // This is the API Key of the Azure OpenAI resource. Optional, uses AzureCliCredential if not provided + }, + "OpenAI": { + "ModelId": "", // This is the Model ID of the OpenAI model + "ApiKey": "" // This is your API Key for the OpenAI service + }, + "UseAzureOpenAI": false // This is a flag to determine whether to use the Azure OpenAI or the OpenAI service + } + ``` + +## QuickStart using Agent Toolkit +1. If you haven't done so already, install the Agents Playground + + ``` + winget install agentsplayground + ``` +1. Start the sample application. +1. Start Agents Playground. At a command prompt: `agentsplayground` + - The tool will open a web browser showing the Microsoft 365 Agents Playground, ready to send messages to your agent. +1. Interact with the Agent via the browser + +## QuickStart using WebChat or Teams + +- Overview of running and testing an Agent + - Provision an Azure Bot in your Azure Subscription + - Configure your Agent settings to use to desired authentication type + - Running an instance of the Agent app (either locally or deployed to Azure) + - Test in a client + +1. Create an Azure Bot with one of these authentication types + - [SingleTenant, Client Secret](https://learn.microsoft.com/en-us/microsoft-365/agents-sdk/azure-bot-create-single-secret) + - [SingleTenant, Federated Credentials](https://learn.microsoft.com/en-us/microsoft-365/agents-sdk/azure-bot-create-federated-credentials) + - [User Assigned Managed Identity](https://learn.microsoft.com/en-us/microsoft-365/agents-sdk/azure-bot-create-managed-identity) + + > Be sure to follow the **Next Steps** at the end of these docs to configure your agent settings. + + > **IMPORTANT:** If you want to run your agent locally via devtunnels, the only support auth type is ClientSecret and Certificates + +1. Running the Agent + 1. Running the Agent locally + - Requires a tunneling tool to allow for local development and debugging should you wish to do local development whilst connected to a external client such as Microsoft Teams. + - **For ClientSecret or Certificate authentication types only.** Federated Credentials and Managed Identity will not work via a tunnel to a local agent and must be deployed to an App Service or container. + + 1. Run `devtunnel`. Please follow [Create and host a dev tunnel](https://learn.microsoft.com/azure/developer/dev-tunnels/get-started?tabs=windows) and host the tunnel with anonymous user access command as shown below: + + ```bash + devtunnel host -p 3978 --allow-anonymous + ``` + + 1. On the Azure Bot, select **Settings**, then **Configuration**, and update the **Messaging endpoint** to `{tunnel-url}/api/messages` + + 1. Start the Agent in Visual Studio + + 1. Deploy Agent code to Azure + 1. VS Publish works well for this. But any tools used to deploy a web application will also work. + 1. On the Azure Bot, select **Settings**, then **Configuration**, and update the **Messaging endpoint** to `https://{{appServiceDomain}}/api/messages` + +## Testing this agent with WebChat + + 1. Select **Test in WebChat** under **Settings** on the Azure Bot in the Azure Portal + +## Testing this Agent in Teams or M365 + +1. Update the manifest.json + - Edit the `manifest.json` contained in the `/appManifest` folder + - Replace with your AppId (that was created above) *everywhere* you see the place holder string `<>` + - Replace `<>` with your Agent url. For example, the tunnel host name. + - Zip up the contents of the `/appManifest` folder to create a `manifest.zip` + - `manifest.json` + - `outline.png` + - `color.png` + +1. Your Azure Bot should have the **Microsoft Teams** channel added under **Channels**. + +1. Navigate to the Microsoft Admin Portal (MAC). Under **Settings** and **Integrated Apps,** select **Upload Custom App**. + +1. Select the `manifest.zip` created in the previous step. + +1. After a short period of time, the agent shows up in Microsoft Teams and Microsoft 365 Copilot. + +## Enabling JWT token validation +1. By default, the AspNet token validation is disabled in order to support local debugging. +1. Enable by updating appsettings + ```json + "TokenValidation": { + "Enabled": true, + "Audiences": [ + "{{ClientId}}" // this is the Client ID used for the Azure Bot + ], + "TenantId": "{{TenantId}}" + }, + ``` + +## Further reading + +To learn more about using the M365 Agent SDK, see [Microsoft 365 Agents SDK](https://learn.microsoft.com/en-us/microsoft-365/agents-sdk/). diff --git a/dotnet/samples/M365Agent/appManifest/color.png b/dotnet/samples/M365Agent/appManifest/color.png new file mode 100644 index 0000000000..b8cf81afbe Binary files /dev/null and b/dotnet/samples/M365Agent/appManifest/color.png differ diff --git a/dotnet/samples/M365Agent/appManifest/manifest.json b/dotnet/samples/M365Agent/appManifest/manifest.json new file mode 100644 index 0000000000..ca5890d8ea --- /dev/null +++ b/dotnet/samples/M365Agent/appManifest/manifest.json @@ -0,0 +1,50 @@ +{ + "$schema": "https://developer.microsoft.com/json-schemas/teams/v1.22/MicrosoftTeams.schema.json", + "manifestVersion": "1.22", + "version": "1.0.0", + "id": "<>", + "developer": { + "name": "Microsoft, Inc.", + "websiteUrl": "https://example.azurewebsites.net", + "privacyUrl": "https://example.azurewebsites.net/privacy", + "termsOfUseUrl": "https://example.azurewebsites.net/termsofuse" + }, + "icons": { + "color": "color.png", + "outline": "outline.png" + }, + "name": { + "short": "AF Sample Agent", + "full": "M365 AgentSDK and Microsoft Agent Framework Sample" + }, + "description": { + "short": "Sample demonstrating M365 AgentSDK, Teams, and Microsoft Agent Framework", + "full": "Sample demonstrating M365 AgentSDK, Teams, and Microsoft Agent Framework" + }, + "accentColor": "#FFFFFF", + "copilotAgents": { + "customEngineAgents": [ + { + "id": "<>", + "type": "bot" + } + ] + }, + "bots": [ + { + "botId": "<>", + "scopes": [ + "personal" + ], + "supportsFiles": false, + "isNotificationOnly": false + } + ], + "permissions": [ + "identity", + "messageTeamMembers" + ], + "validDomains": [ + "<>" + ] +} \ No newline at end of file diff --git a/dotnet/samples/M365Agent/appManifest/outline.png b/dotnet/samples/M365Agent/appManifest/outline.png new file mode 100644 index 0000000000..2c3bf6fa65 Binary files /dev/null and b/dotnet/samples/M365Agent/appManifest/outline.png differ diff --git a/dotnet/samples/M365Agent/appsettings.json.template b/dotnet/samples/M365Agent/appsettings.json.template new file mode 100644 index 0000000000..7268acf39b --- /dev/null +++ b/dotnet/samples/M365Agent/appsettings.json.template @@ -0,0 +1,54 @@ +{ + "TokenValidation": { + "Enabled": false, + "Audiences": [ + "{{ClientId}}" // this is the Client ID used for the Azure Bot + ], + "TenantId": "{{TenantId}}" + }, + + "AgentApplication": { + "StartTypingTimer": true, + "RemoveRecipientMention": false, + "NormalizeMentions": false + }, + + "Connections": { + "ServiceConnection": { + "Settings": { + // this is the AuthType for the connection, valid values can be found in Microsoft.Agents.Authentication.Msal.Model.AuthTypes. The default is ClientSecret. + "AuthType": "" + + // Other properties dependent on the authorization type the Azure Bot uses. + } + } + }, + "ConnectionsMap": [ + { + "ServiceUrl": "*", + "Connection": "ServiceConnection" + } + ], + + // This is the configuration for the AI services, use environment variables or user secrets to store sensitive information. + // Do not store sensitive information in this file + "AIServices": { + "AzureOpenAI": { + "DeploymentName": "", // This is the Deployment (as opposed to model) Name of the Azure OpenAI model + "Endpoint": "", // This is the Endpoint of the Azure OpenAI resource + "ApiKey": "" // This is the API Key of the Azure OpenAI resource. Optional, uses AzureCliCredential if not provided + }, + "OpenAI": { + "ModelId": "", // This is the Model ID of the OpenAI model + "ApiKey": "" // This is your API Key for the OpenAI service + }, + "UseAzureOpenAI": false // This is a flag to determine whether to use the Azure OpenAI or the OpenAI service + }, + + "Logging": { + "LogLevel": { + "Default": "Information", + "Microsoft.AspNetCore": "Warning" + } + } +} \ No newline at end of file diff --git a/dotnet/samples/Catalog/AgentWithTextSearchRag/AgentWithTextSearchRag.csproj b/dotnet/samples/Purview/AgentWithPurview/AgentWithPurview.csproj similarity index 68% rename from dotnet/samples/Catalog/AgentWithTextSearchRag/AgentWithTextSearchRag.csproj rename to dotnet/samples/Purview/AgentWithPurview/AgentWithPurview.csproj index c6bab8327e..0a79857d64 100644 --- a/dotnet/samples/Catalog/AgentWithTextSearchRag/AgentWithTextSearchRag.csproj +++ b/dotnet/samples/Purview/AgentWithPurview/AgentWithPurview.csproj @@ -1,8 +1,8 @@ - + Exe - net9.0 + net10.0 enable enable @@ -11,11 +11,11 @@ - + diff --git a/dotnet/samples/Purview/AgentWithPurview/Program.cs b/dotnet/samples/Purview/AgentWithPurview/Program.cs new file mode 100644 index 0000000000..842917b427 --- /dev/null +++ b/dotnet/samples/Purview/AgentWithPurview/Program.cs @@ -0,0 +1,43 @@ +// Copyright (c) Microsoft. All rights reserved. + +// This sample shows how to create and use a simple AI agent with Purview integration. +// It uses Azure OpenAI as the backend, but any IChatClient can be used. +// Authentication to Purview is done using an InteractiveBrowserCredential. +// Any TokenCredential with Purview API permissions can be used here. + +using Azure.AI.OpenAI; +using Azure.Core; +using Azure.Identity; +using Microsoft.Agents.AI; +using Microsoft.Agents.AI.Purview; +using Microsoft.Extensions.AI; + +var endpoint = Environment.GetEnvironmentVariable("AZURE_OPENAI_ENDPOINT") ?? throw new InvalidOperationException("AZURE_OPENAI_ENDPOINT is not set."); +var deploymentName = Environment.GetEnvironmentVariable("AZURE_OPENAI_DEPLOYMENT_NAME") ?? "gpt-4o-mini"; +var purviewClientAppId = Environment.GetEnvironmentVariable("PURVIEW_CLIENT_APP_ID") ?? throw new InvalidOperationException("PURVIEW_CLIENT_APP_ID is not set."); + +// This will get a user token for an entra app configured to call the Purview API. +// Any TokenCredential with permissions to call the Purview API can be used here. +TokenCredential browserCredential = new InteractiveBrowserCredential( + new InteractiveBrowserCredentialOptions + { + ClientId = purviewClientAppId + }); + +using IChatClient client = new AzureOpenAIClient( + new Uri(endpoint), + new AzureCliCredential()) + .GetOpenAIResponseClient(deploymentName) + .AsIChatClient() + .AsBuilder() + .WithPurview(browserCredential, new PurviewSettings("Agent Framework Test App")) + .Build(); + +Console.WriteLine("Enter a prompt to send to the client:"); +string? promptText = Console.ReadLine(); + +if (!string.IsNullOrEmpty(promptText)) +{ + // Invoke the agent and output the text result. + Console.WriteLine(await client.GetResponseAsync(promptText)); +} diff --git a/dotnet/samples/README.md b/dotnet/samples/README.md index d6f2f5c39c..db0202794a 100644 --- a/dotnet/samples/README.md +++ b/dotnet/samples/README.md @@ -19,6 +19,7 @@ The samples are subdivided into the following categories: - [Getting Started - Agent Providers](./GettingStarted/AgentProviders/README.md): Shows how to create an AIAgent instance for a selection of providers. - [Getting Started - Agent Telemetry](./GettingStarted/AgentOpenTelemetry/README.md): Demo which showcases the integration of OpenTelemetry with the Microsoft Agent Framework using Azure OpenAI and .NET Aspire Dashboard for telemetry visualization. - [Semantic Kernel to Agent Framework Migration](https://github.com/microsoft/semantic-kernel/tree/main/dotnet/samples/AgentFrameworkMigration): For instructions and samples describing how to migrate from Semantic Kernel to Microsoft Agent Framework +- [Azure Functions](./AzureFunctions/README.md): Samples for using the Microsoft Agent Framework with Azure Functions via the durable task extension. ## Prerequisites diff --git a/dotnet/src/Microsoft.Agents.AI.A2A/A2AAgent.cs b/dotnet/src/Microsoft.Agents.AI.A2A/A2AAgent.cs index cafbf90b87..e804fbb389 100644 --- a/dotnet/src/Microsoft.Agents.AI.A2A/A2AAgent.cs +++ b/dotnet/src/Microsoft.Agents.AI.A2A/A2AAgent.cs @@ -3,6 +3,7 @@ using System; using System.Collections.Generic; using System.Linq; +using System.Net.ServerSentEvents; using System.Runtime.CompilerServices; using System.Text.Json; using System.Threading; @@ -74,26 +75,28 @@ public override async Task RunAsync(IEnumerable m { _ = Throw.IfNull(messages); - var a2aMessage = messages.ToA2AMessage(); + A2AAgentThread typedThread = this.GetA2AThread(thread, options); - thread ??= this.GetNewThread(); - if (thread is not A2AAgentThread typedThread) - { - throw new InvalidOperationException("The provided thread is not compatible with the agent. Only threads created by the agent can be used."); - } + this._logger.LogA2AAgentInvokingAgent(nameof(RunAsync), this.Id, this.Name); - // Linking the message to the existing conversation, if any. - a2aMessage.ContextId = typedThread.ContextId; + A2AResponse? a2aResponse = null; - this._logger.LogA2AAgentInvokingAgent(nameof(RunAsync), this.Id, this.Name); + if (GetContinuationToken(messages, options) is { } token) + { + a2aResponse = await this._a2aClient.GetTaskAsync(token.TaskId, cancellationToken).ConfigureAwait(false); + } + else + { + var a2aMessage = CreateA2AMessage(typedThread, messages); - var a2aResponse = await this._a2aClient.SendMessageAsync(new MessageSendParams { Message = a2aMessage }, cancellationToken).ConfigureAwait(false); + a2aResponse = await this._a2aClient.SendMessageAsync(new MessageSendParams { Message = a2aMessage }, cancellationToken).ConfigureAwait(false); + } this._logger.LogAgentChatClientInvokedAgent(nameof(RunAsync), this.Id, this.Name); if (a2aResponse is AgentMessage message) { - UpdateThreadConversationId(typedThread, message.ContextId); + UpdateThread(typedThread, message.ContextId); return new AgentRunResponse { @@ -101,21 +104,30 @@ public override async Task RunAsync(IEnumerable m ResponseId = message.MessageId, RawRepresentation = message, Messages = [message.ToChatMessage()], - AdditionalProperties = message.Metadata.ToAdditionalProperties(), + AdditionalProperties = message.Metadata?.ToAdditionalProperties(), }; } + if (a2aResponse is AgentTask agentTask) { - UpdateThreadConversationId(typedThread, agentTask.ContextId); + UpdateThread(typedThread, agentTask.ContextId, agentTask.Id); - return new AgentRunResponse + var response = new AgentRunResponse { AgentId = this.Id, ResponseId = agentTask.Id, RawRepresentation = agentTask, - Messages = agentTask.ToChatMessages(), - AdditionalProperties = agentTask.Metadata.ToAdditionalProperties(), + Messages = agentTask.ToChatMessages() ?? [], + ContinuationToken = CreateContinuationToken(agentTask.Id, agentTask.Status.State), + AdditionalProperties = agentTask.Metadata?.ToAdditionalProperties(), }; + + if (agentTask.ToChatMessages() is { Count: > 0 } taskMessages) + { + response.Messages = taskMessages; + } + + return response; } throw new NotSupportedException($"Only Message and AgentTask responses are supported from A2A agents. Received: {a2aResponse.GetType().FullName ?? "null"}"); @@ -126,47 +138,67 @@ public override async IAsyncEnumerable RunStreamingAsync { _ = Throw.IfNull(messages); - var a2aMessage = messages.ToA2AMessage(); + A2AAgentThread typedThread = this.GetA2AThread(thread, options); - thread ??= this.GetNewThread(); - if (thread is not A2AAgentThread typedThread) + this._logger.LogA2AAgentInvokingAgent(nameof(RunStreamingAsync), this.Id, this.Name); + + ConfiguredCancelableAsyncEnumerable> a2aSseEvents; + + if (options?.ContinuationToken is not null) { - throw new InvalidOperationException("The provided thread is not compatible with the agent. Only threads created by the agent can be used."); + // Task stream resumption is not well defined in the A2A v2.* specification, leaving it to the agent implementations. + // The v3.0 specification improves this by defining task stream reconnection that allows obtaining the same stream + // from the beginning, but it does not define stream resumption from a specific point in the stream. + // Therefore, the code should be updated once the A2A .NET library supports the A2A v3.0 specification, + // and AF has the necessary model to allow consumers to know whether they need to resume the stream and add new updates to + // the existing ones or reconnect the stream and obtain all updates again. + // For more details, see the following issue: https://github.com/microsoft/agent-framework/issues/1764 + throw new InvalidOperationException("Reconnecting to task streams using continuation tokens is not supported yet."); + // a2aSseEvents = this._a2aClient.SubscribeToTaskAsync(token.TaskId, cancellationToken).ConfigureAwait(false); } - // Linking the message to the existing conversation, if any. - a2aMessage.ContextId = typedThread.ContextId; - - this._logger.LogA2AAgentInvokingAgent(nameof(RunStreamingAsync), this.Id, this.Name); + var a2aMessage = CreateA2AMessage(typedThread, messages); - var a2aSseEvents = this._a2aClient.SendMessageStreamingAsync(new MessageSendParams { Message = a2aMessage }, cancellationToken).ConfigureAwait(false); + a2aSseEvents = this._a2aClient.SendMessageStreamingAsync(new MessageSendParams { Message = a2aMessage }, cancellationToken).ConfigureAwait(false); this._logger.LogAgentChatClientInvokedAgent(nameof(RunStreamingAsync), this.Id, this.Name); + string? contextId = null; + string? taskId = null; + await foreach (var sseEvent in a2aSseEvents) { - if (sseEvent.Data is not AgentMessage message) + if (sseEvent.Data is AgentMessage message) { - throw new NotSupportedException($"Only message responses are supported from A2A agents. Received: {sseEvent.Data?.GetType().FullName ?? "null"}"); + contextId = message.ContextId; + + yield return this.ConvertToAgentResponseUpdate(message); } + else if (sseEvent.Data is AgentTask task) + { + contextId = task.ContextId; + taskId = task.Id; - UpdateThreadConversationId(typedThread, message.ContextId); + yield return this.ConvertToAgentResponseUpdate(task); + } + else if (sseEvent.Data is TaskUpdateEvent taskUpdateEvent) + { + contextId = taskUpdateEvent.ContextId; + taskId = taskUpdateEvent.TaskId; - yield return new AgentRunResponseUpdate + yield return this.ConvertToAgentResponseUpdate(taskUpdateEvent); + } + else { - AgentId = this.Id, - ResponseId = message.MessageId, - RawRepresentation = message, - Role = ChatRole.Assistant, - MessageId = message.MessageId, - Contents = [.. message.Parts.Select(part => part.ToAIContent()).OfType()], - AdditionalProperties = message.Metadata.ToAdditionalProperties(), - }; + throw new NotSupportedException($"Only message, task, task update events are supported from A2A agents. Received: {sseEvent.Data.GetType().FullName ?? "null"}"); + } } + + UpdateThread(typedThread, contextId, taskId); } /// - public override string Id => this._id ?? base.Id; + protected override string? IdCore => this._id; /// public override string? Name => this._name ?? base.Name; @@ -177,7 +209,27 @@ public override async IAsyncEnumerable RunStreamingAsync /// public override string? Description => this._description ?? base.Description; - private static void UpdateThreadConversationId(A2AAgentThread? thread, string? contextId) + private A2AAgentThread GetA2AThread(AgentThread? thread, AgentRunOptions? options) + { + // Aligning with other agent implementations that support background responses, where + // a thread is required for background responses to prevent inconsistent experience + // for callers if they forget to provide the thread for initial or follow-up runs. + if (options?.AllowBackgroundResponses is true && thread is null) + { + throw new InvalidOperationException("A thread must be provided when AllowBackgroundResponses is enabled."); + } + + thread ??= this.GetNewThread(); + + if (thread is not A2AAgentThread typedThread) + { + throw new InvalidOperationException($"The provided thread type {thread.GetType()} is not compatible with the agent. Only A2A agent created threads are supported."); + } + + return typedThread; + } + + private static void UpdateThread(A2AAgentThread? thread, string? contextId, string? taskId = null) { if (thread is null) { @@ -194,5 +246,93 @@ private static void UpdateThreadConversationId(A2AAgentThread? thread, string? c // Assign a server-generated context Id to the thread if it's not already set. thread.ContextId ??= contextId; + thread.TaskId = taskId; + } + + private static AgentMessage CreateA2AMessage(A2AAgentThread typedThread, IEnumerable messages) + { + var a2aMessage = messages.ToA2AMessage(); + + // Linking the message to the existing conversation, if any. + // See: https://github.com/a2aproject/A2A/blob/main/docs/topics/life-of-a-task.md#group-related-interactions + a2aMessage.ContextId = typedThread.ContextId; + + // Link the message as a follow-up to an existing task, if any. + // See: https://github.com/a2aproject/A2A/blob/main/docs/topics/life-of-a-task.md#task-refinements + a2aMessage.ReferenceTaskIds = typedThread.TaskId is null ? null : [typedThread.TaskId]; + + return a2aMessage; + } + + private static A2AContinuationToken? GetContinuationToken(IEnumerable messages, AgentRunOptions? options = null) + { + if (options?.ContinuationToken is ResponseContinuationToken token) + { + if (messages.Any()) + { + throw new InvalidOperationException("Messages are not allowed when continuing a background response using a continuation token."); + } + + return A2AContinuationToken.FromToken(token); + } + + return null; + } + + private static A2AContinuationToken? CreateContinuationToken(string taskId, TaskState state) + { + if (state is TaskState.Submitted or TaskState.Working) + { + return new A2AContinuationToken(taskId); + } + + return null; + } + + private AgentRunResponseUpdate ConvertToAgentResponseUpdate(AgentMessage message) + { + return new AgentRunResponseUpdate + { + AgentId = this.Id, + ResponseId = message.MessageId, + RawRepresentation = message, + Role = ChatRole.Assistant, + MessageId = message.MessageId, + Contents = message.Parts.ConvertAll(part => part.ToAIContent()), + AdditionalProperties = message.Metadata?.ToAdditionalProperties(), + }; + } + + private AgentRunResponseUpdate ConvertToAgentResponseUpdate(AgentTask task) + { + return new AgentRunResponseUpdate + { + AgentId = this.Id, + ResponseId = task.Id, + RawRepresentation = task, + Role = ChatRole.Assistant, + Contents = task.ToAIContents(), + AdditionalProperties = task.Metadata?.ToAdditionalProperties(), + }; + } + + private AgentRunResponseUpdate ConvertToAgentResponseUpdate(TaskUpdateEvent taskUpdateEvent) + { + AgentRunResponseUpdate responseUpdate = new() + { + AgentId = this.Id, + ResponseId = taskUpdateEvent.TaskId, + RawRepresentation = taskUpdateEvent, + Role = ChatRole.Assistant, + AdditionalProperties = taskUpdateEvent.Metadata?.ToAdditionalProperties() ?? [], + }; + + if (taskUpdateEvent is TaskArtifactUpdateEvent artifactUpdateEvent) + { + responseUpdate.Contents = artifactUpdateEvent.Artifact.ToAIContents(); + responseUpdate.RawRepresentation = artifactUpdateEvent; + } + + return responseUpdate; } } diff --git a/dotnet/src/Microsoft.Agents.AI.A2A/A2AAgentThread.cs b/dotnet/src/Microsoft.Agents.AI.A2A/A2AAgentThread.cs index 010df78a02..55942c8dd1 100644 --- a/dotnet/src/Microsoft.Agents.AI.A2A/A2AAgentThread.cs +++ b/dotnet/src/Microsoft.Agents.AI.A2A/A2AAgentThread.cs @@ -1,5 +1,6 @@ // Copyright (c) Microsoft. All rights reserved. +using System; using System.Text.Json; namespace Microsoft.Agents.AI.A2A; @@ -7,22 +8,59 @@ namespace Microsoft.Agents.AI.A2A; /// /// Thread for A2A based agents. /// -public sealed class A2AAgentThread : ServiceIdAgentThread +public sealed class A2AAgentThread : AgentThread { internal A2AAgentThread() { } - internal A2AAgentThread(JsonElement serializedThreadState, JsonSerializerOptions? jsonSerializerOptions = null) : base(serializedThreadState, jsonSerializerOptions) + internal A2AAgentThread(JsonElement serializedThreadState, JsonSerializerOptions? jsonSerializerOptions = null) { + if (serializedThreadState.ValueKind != JsonValueKind.Object) + { + throw new ArgumentException("The serialized thread state must be a JSON object.", nameof(serializedThreadState)); + } + + var state = serializedThreadState.Deserialize( + A2AJsonUtilities.DefaultOptions.GetTypeInfo(typeof(A2AAgentThreadState))) as A2AAgentThreadState; + + if (state?.ContextId is string contextId) + { + this.ContextId = contextId; + } + + if (state?.TaskId is string taskId) + { + this.TaskId = taskId; + } } /// /// Gets the ID for the current conversation with the A2A agent. /// - public string? ContextId + public string? ContextId { get; internal set; } + + /// + /// Gets the ID for the task the agent is currently working on. + /// + public string? TaskId { get; internal set; } + + /// + public override JsonElement Serialize(JsonSerializerOptions? jsonSerializerOptions = null) + { + var state = new A2AAgentThreadState + { + ContextId = this.ContextId, + TaskId = this.TaskId + }; + + return JsonSerializer.SerializeToElement(state, A2AJsonUtilities.DefaultOptions.GetTypeInfo(typeof(A2AAgentThreadState))); + } + + internal sealed class A2AAgentThreadState { - get { return this.ServiceThreadId; } - internal set { this.ServiceThreadId = value; } + public string? ContextId { get; set; } + + public string? TaskId { get; set; } } } diff --git a/dotnet/src/Microsoft.Agents.AI.A2A/A2AContinuationToken.cs b/dotnet/src/Microsoft.Agents.AI.A2A/A2AContinuationToken.cs new file mode 100644 index 0000000000..5233adb88f --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.A2A/A2AContinuationToken.cs @@ -0,0 +1,81 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.IO; +using System.Text.Json; +using Microsoft.Extensions.AI; +using Microsoft.Shared.Diagnostics; + +namespace Microsoft.Agents.AI.A2A; +#pragma warning disable MEAI001 // Type is for evaluation purposes only and is subject to change or removal in future updates. Suppress this diagnostic to proceed. +internal class A2AContinuationToken : ResponseContinuationToken +{ + internal A2AContinuationToken(string taskId) + { + _ = Throw.IfNullOrEmpty(taskId); + + this.TaskId = taskId; + } + + internal string TaskId { get; } + + internal static A2AContinuationToken FromToken(ResponseContinuationToken token) + { + if (token is A2AContinuationToken longRunContinuationToken) + { + return longRunContinuationToken; + } + + ReadOnlyMemory data = token.ToBytes(); + + if (data.Length == 0) + { + Throw.ArgumentException(nameof(token), "Failed to create A2AContinuationToken from provided token because it does not contain any data."); + } + + Utf8JsonReader reader = new(data.Span); + + string taskId = null!; + + reader.Read(); + + while (reader.Read()) + { + if (reader.TokenType == JsonTokenType.EndObject) + { + break; + } + + string propertyName = reader.GetString() ?? throw new JsonException("Failed to read property name from continuation token."); + + switch (propertyName) + { + case "taskId": + reader.Read(); + taskId = reader.GetString()!; + break; + default: + throw new JsonException($"Unrecognized property '{propertyName}'."); + } + } + + return new(taskId); + } + + public override ReadOnlyMemory ToBytes() + { + using MemoryStream stream = new(); + using Utf8JsonWriter writer = new(stream); + + writer.WriteStartObject(); + + writer.WriteString("taskId", this.TaskId); + + writer.WriteEndObject(); + + writer.Flush(); + stream.Position = 0; + + return stream.ToArray(); + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.A2A/A2AJsonUtilities.cs b/dotnet/src/Microsoft.Agents.AI.A2A/A2AJsonUtilities.cs new file mode 100644 index 0000000000..2fbb2e8617 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.A2A/A2AJsonUtilities.cs @@ -0,0 +1,80 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Diagnostics.CodeAnalysis; +using System.Text.Encodings.Web; +using System.Text.Json; +using System.Text.Json.Serialization; +using Microsoft.Agents.AI.A2A; + +namespace Microsoft.Agents.AI; + +/// +/// Provides utility methods and configurations for JSON serialization operations for A2A agent types. +/// +public static partial class A2AJsonUtilities +{ + /// + /// Gets the default instance used for JSON serialization operations of A2A agent types. + /// + /// + /// + /// For Native AOT or applications disabling , this instance + /// includes source generated contracts for A2A agent types. + /// + /// + /// It additionally turns on the following settings: + /// + /// Enables defaults. + /// Enables as the default ignore condition for properties. + /// Enables as the default number handling for number types. + /// + /// Enables when escaping JSON strings. + /// Consuming applications must ensure that JSON outputs are adequately escaped before embedding in other document formats, such as HTML and XML. + /// + /// + /// + /// + public static JsonSerializerOptions DefaultOptions { get; } = CreateDefaultOptions(); + + /// + /// Creates and configures the default JSON serialization options for agent abstraction types. + /// + /// The configured options. + [UnconditionalSuppressMessage("ReflectionAnalysis", "IL3050:RequiresDynamicCode", Justification = "Converter is guarded by IsReflectionEnabledByDefault check.")] + [UnconditionalSuppressMessage("Trimming", "IL2026:Members annotated with 'RequiresUnreferencedCodeAttribute' require dynamic access", Justification = "Converter is guarded by IsReflectionEnabledByDefault check.")] + private static JsonSerializerOptions CreateDefaultOptions() + { + // Copy the configuration from the source generated context. + JsonSerializerOptions options = new(JsonContext.Default.Options) + { + Encoder = JavaScriptEncoder.UnsafeRelaxedJsonEscaping, // same as AIJsonUtilities + }; + + // Chain in the resolvers from both AIJsonUtilities and our source generated context. + // We want AIJsonUtilities first to ensure any M.E.AI types are handled via its resolver. + options.TypeInfoResolverChain.Clear(); + options.TypeInfoResolverChain.Add(AgentAbstractionsJsonUtilities.DefaultOptions.TypeInfoResolver!); + + // If reflection-based serialization is enabled by default, this includes + // the default type info resolver that utilizes reflection, but we need to manually + // apply the same converter AIJsonUtilities adds for string-based enum serialization, + // as that's not propagated as part of the resolver. + if (JsonSerializer.IsReflectionEnabledByDefault) + { + options.Converters.Add(new JsonStringEnumConverter()); + } + + options.MakeReadOnly(); + return options; + } + + [JsonSourceGenerationOptions(JsonSerializerDefaults.Web, + UseStringEnumConverter = true, + DefaultIgnoreCondition = JsonIgnoreCondition.WhenWritingNull, + NumberHandling = JsonNumberHandling.AllowReadingFromString)] + + // A2A agent types + [JsonSerializable(typeof(A2AAgentThread.A2AAgentThreadState))] + [ExcludeFromCodeCoverage] + private sealed partial class JsonContext : JsonSerializerContext; +} diff --git a/dotnet/src/Microsoft.Agents.AI.A2A/Extensions/A2AAgentTaskExtensions.cs b/dotnet/src/Microsoft.Agents.AI.A2A/Extensions/A2AAgentTaskExtensions.cs index 236ecfb174..a577ad9364 100644 --- a/dotnet/src/Microsoft.Agents.AI.A2A/Extensions/A2AAgentTaskExtensions.cs +++ b/dotnet/src/Microsoft.Agents.AI.A2A/Extensions/A2AAgentTaskExtensions.cs @@ -11,20 +11,37 @@ namespace A2A; /// internal static class A2AAgentTaskExtensions { - internal static IList ToChatMessages(this AgentTask agentTask) + internal static IList? ToChatMessages(this AgentTask agentTask) { _ = Throw.IfNull(agentTask); - List messages = []; + List? messages = null; - if (agentTask.Artifacts is not null) + if (agentTask?.Artifacts is { Count: > 0 }) { foreach (var artifact in agentTask.Artifacts) { - messages.Add(artifact.ToChatMessage()); + (messages ??= []).Add(artifact.ToChatMessage()); } } return messages; } + + internal static IList? ToAIContents(this AgentTask agentTask) + { + _ = Throw.IfNull(agentTask); + + List? aiContents = null; + + if (agentTask.Artifacts is not null) + { + foreach (var artifact in agentTask.Artifacts) + { + (aiContents ??= []).AddRange(artifact.ToAIContents()); + } + } + + return aiContents; + } } diff --git a/dotnet/src/Microsoft.Agents.AI.A2A/Extensions/A2AArtifactExtensions.cs b/dotnet/src/Microsoft.Agents.AI.A2A/Extensions/A2AArtifactExtensions.cs index 36683d549b..cecd9a8504 100644 --- a/dotnet/src/Microsoft.Agents.AI.A2A/Extensions/A2AArtifactExtensions.cs +++ b/dotnet/src/Microsoft.Agents.AI.A2A/Extensions/A2AArtifactExtensions.cs @@ -12,21 +12,15 @@ internal static class A2AArtifactExtensions { internal static ChatMessage ToChatMessage(this Artifact artifact) { - List? aiContents = null; - - foreach (var part in artifact.Parts) - { - var content = part.ToAIContent(); - if (content is not null) - { - (aiContents ??= []).Add(content); - } - } - - return new ChatMessage(ChatRole.Assistant, aiContents) + return new ChatMessage(ChatRole.Assistant, artifact.ToAIContents()) { AdditionalProperties = artifact.Metadata.ToAdditionalProperties(), RawRepresentation = artifact, }; } + + internal static List ToAIContents(this Artifact artifact) + { + return artifact.Parts.ConvertAll(part => part.ToAIContent()); + } } diff --git a/dotnet/src/Microsoft.Agents.AI.A2A/Microsoft.Agents.AI.A2A.csproj b/dotnet/src/Microsoft.Agents.AI.A2A/Microsoft.Agents.AI.A2A.csproj index 46e3c97d8f..b1b9ba7671 100644 --- a/dotnet/src/Microsoft.Agents.AI.A2A/Microsoft.Agents.AI.A2A.csproj +++ b/dotnet/src/Microsoft.Agents.AI.A2A/Microsoft.Agents.AI.A2A.csproj @@ -1,21 +1,19 @@ - $(ProjectsTargetFrameworks) - $(ProjectsDebugTargetFrameworks) preview + $(NoWarn);MEAI001 true + true - - diff --git a/dotnet/src/Microsoft.Agents.AI.AGUI/AGUIAgent.cs b/dotnet/src/Microsoft.Agents.AI.AGUI/AGUIAgent.cs deleted file mode 100644 index e86fac7429..0000000000 --- a/dotnet/src/Microsoft.Agents.AI.AGUI/AGUIAgent.cs +++ /dev/null @@ -1,102 +0,0 @@ -// Copyright (c) Microsoft. All rights reserved. - -using System; -using System.Collections.Generic; -using System.Linq; -using System.Net.Http; -using System.Runtime.CompilerServices; -using System.Text.Json; -using System.Threading; -using System.Threading.Tasks; -using Microsoft.Agents.AI.AGUI.Shared; -using Microsoft.Extensions.AI; -using Microsoft.Shared.Diagnostics; - -namespace Microsoft.Agents.AI.AGUI; - -/// -/// Provides an implementation that communicates with an AG-UI compliant server. -/// -public sealed class AGUIAgent : AIAgent -{ - private readonly AGUIHttpService _client; - - /// - /// Initializes a new instance of the class. - /// - /// The agent ID. - /// Optional description of the agent. - /// The HTTP client to use for communication with the AG-UI server. - /// The URL for the AG-UI server. - public AGUIAgent(string id, string description, HttpClient httpClient, string endpoint) - { - this.Id = Throw.IfNullOrWhitespace(id); - this.Description = description; - this._client = new AGUIHttpService( - httpClient ?? Throw.IfNull(httpClient), - endpoint ?? Throw.IfNullOrEmpty(endpoint)); - } - - /// - public override string Id { get; } - - /// - public override string? Description { get; } - - /// - public override AgentThread GetNewThread() => new AGUIAgentThread(); - - /// - public override AgentThread DeserializeThread(JsonElement serializedThread, JsonSerializerOptions? jsonSerializerOptions = null) => - new AGUIAgentThread(serializedThread, jsonSerializerOptions); - - /// - public override async Task RunAsync( - IEnumerable messages, - AgentThread? thread = null, - AgentRunOptions? options = null, - CancellationToken cancellationToken = default) - { - return await this.RunStreamingAsync(messages, thread, null, cancellationToken) - .ToAgentRunResponseAsync(cancellationToken) - .ConfigureAwait(false); - } - - /// - public override async IAsyncEnumerable RunStreamingAsync( - IEnumerable messages, - AgentThread? thread = null, - AgentRunOptions? options = null, - [EnumeratorCancellation] CancellationToken cancellationToken = default) - { - List updates = []; - - _ = Throw.IfNull(messages); - - if ((thread ?? this.GetNewThread()) is not AGUIAgentThread typedThread) - { - throw new InvalidOperationException("The provided thread is not compatible with the agent. Only threads created by the agent can be used."); - } - - string runId = $"run_{Guid.NewGuid()}"; - - var llmMessages = typedThread.MessageStore.Concat(messages); - - RunAgentInput input = new() - { - ThreadId = typedThread.ThreadId, - RunId = runId, - Messages = llmMessages.AsAGUIMessages(), - }; - - await foreach (var update in this._client.PostRunAsync(input, cancellationToken).AsAgentRunResponseUpdatesAsync(cancellationToken).ConfigureAwait(false)) - { - ChatResponseUpdate chatUpdate = update.AsChatResponseUpdate(); - updates.Add(chatUpdate); - yield return update; - } - - ChatResponse response = updates.ToChatResponse(); - await NotifyThreadOfNewMessagesAsync(typedThread, messages.Concat(response.Messages), cancellationToken).ConfigureAwait(false); - } -} diff --git a/dotnet/src/Microsoft.Agents.AI.AGUI/AGUIAgentThread.cs b/dotnet/src/Microsoft.Agents.AI.AGUI/AGUIAgentThread.cs deleted file mode 100644 index 5b2f29897a..0000000000 --- a/dotnet/src/Microsoft.Agents.AI.AGUI/AGUIAgentThread.cs +++ /dev/null @@ -1,61 +0,0 @@ -// Copyright (c) Microsoft. All rights reserved. - -using System; -using System.Text.Json; -using Microsoft.Shared.Diagnostics; - -namespace Microsoft.Agents.AI.AGUI; - -internal sealed class AGUIAgentThread : InMemoryAgentThread -{ - public AGUIAgentThread() - : base() - { - this.ThreadId = Guid.NewGuid().ToString(); - } - - public AGUIAgentThread(JsonElement serializedThreadState, JsonSerializerOptions? jsonSerializerOptions = null) - : base(UnwrapState(serializedThreadState), jsonSerializerOptions) - { - var threadId = serializedThreadState.TryGetProperty(nameof(AGUIAgentThreadState.ThreadId), out var stateElement) - ? stateElement.GetString() - : null; - - if (string.IsNullOrEmpty(threadId)) - { - Throw.InvalidOperationException("Serialized thread is missing required ThreadId."); - } - this.ThreadId = threadId; - } - - private static JsonElement UnwrapState(JsonElement serializedThreadState) - { - var state = serializedThreadState.Deserialize(AGUIJsonSerializerContext.Default.AGUIAgentThreadState); - if (state == null) - { - Throw.InvalidOperationException("Serialized thread is missing required WrappedState."); - } - - return state.WrappedState; - } - - public string ThreadId { get; set; } - - public override JsonElement Serialize(JsonSerializerOptions? jsonSerializerOptions = null) - { - var wrappedState = base.Serialize(jsonSerializerOptions); - var state = new AGUIAgentThreadState - { - ThreadId = this.ThreadId, - WrappedState = wrappedState, - }; - - return JsonSerializer.SerializeToElement(state, AGUIJsonSerializerContext.Default.AGUIAgentThreadState); - } - - internal sealed class AGUIAgentThreadState - { - public string ThreadId { get; set; } = string.Empty; - public JsonElement WrappedState { get; set; } - } -} diff --git a/dotnet/src/Microsoft.Agents.AI.AGUI/AGUIChatClient.cs b/dotnet/src/Microsoft.Agents.AI.AGUI/AGUIChatClient.cs new file mode 100644 index 0000000000..37c9c60413 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.AGUI/AGUIChatClient.cs @@ -0,0 +1,379 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Collections.Generic; +using System.Linq; +using System.Net.Http; +using System.Net.Http.Headers; +using System.Runtime.CompilerServices; +using System.Text.Json; +using System.Threading; +using System.Threading.Tasks; +using Microsoft.Agents.AI.AGUI.Shared; +using Microsoft.Extensions.AI; +using Microsoft.Extensions.Logging; +using Microsoft.Extensions.Logging.Abstractions; +using Microsoft.Shared.Diagnostics; + +namespace Microsoft.Agents.AI.AGUI; + +/// +/// Provides an implementation that communicates with an AG-UI compliant server. +/// +public sealed class AGUIChatClient : DelegatingChatClient +{ + /// + /// Initializes a new instance of the class. + /// + /// The HTTP client to use for communication with the AG-UI server. + /// The URL for the AG-UI server. + /// The to use for logging. + /// JSON serializer options for tool call argument serialization. If null, AGUIJsonSerializerContext.Default.Options will be used. + /// Optional service provider for resolving dependencies like ILogger. + public AGUIChatClient( + HttpClient httpClient, + string endpoint, + ILoggerFactory? loggerFactory = null, + JsonSerializerOptions? jsonSerializerOptions = null, + IServiceProvider? serviceProvider = null) : base(CreateInnerClient( + httpClient, + endpoint, + CombineJsonSerializerOptions(jsonSerializerOptions), + loggerFactory, + serviceProvider)) + { + } + + private static JsonSerializerOptions CombineJsonSerializerOptions(JsonSerializerOptions? jsonSerializerOptions) + { + if (jsonSerializerOptions == null) + { + return AGUIJsonSerializerContext.Default.Options; + } + + // Create a new JsonSerializerOptions based on the provided one + var combinedOptions = new JsonSerializerOptions(jsonSerializerOptions); + + // Add the AGUI context to the type info resolver chain if not already present + if (!combinedOptions.TypeInfoResolverChain.Any(r => r == AGUIJsonSerializerContext.Default)) + { + combinedOptions.TypeInfoResolverChain.Insert(0, AGUIJsonSerializerContext.Default); + } + + return combinedOptions; + } + + private static FunctionInvokingChatClient CreateInnerClient( + HttpClient httpClient, + string endpoint, + JsonSerializerOptions jsonSerializerOptions, + ILoggerFactory? loggerFactory, + IServiceProvider? serviceProvider) + { + Throw.IfNull(httpClient); + Throw.IfNull(endpoint); + var handler = new AGUIChatClientHandler(httpClient, endpoint, jsonSerializerOptions, serviceProvider); + return new FunctionInvokingChatClient(handler, loggerFactory, serviceProvider); + } + + /// + public override Task GetResponseAsync(IEnumerable messages, ChatOptions? options = null, CancellationToken cancellationToken = default) => + this.GetStreamingResponseAsync(messages, options, cancellationToken) + .ToChatResponseAsync(cancellationToken); + + /// + public override async IAsyncEnumerable GetStreamingResponseAsync( + IEnumerable messages, + ChatOptions? options = null, + [EnumeratorCancellation] CancellationToken cancellationToken = default) + { + ChatResponseUpdate? firstUpdate = null; + string? conversationId = null; + // AG-UI requires the full message history on every turn, so we clear the conversation id here + // and restore it for the caller. + var innerOptions = options; + if (options?.ConversationId != null) + { + conversationId = options.ConversationId; + + // Clone the options and set the conversation ID to null so the FunctionInvokingChatClient doesn't see it. + innerOptions = options.Clone(); + innerOptions.AdditionalProperties ??= []; + innerOptions.AdditionalProperties["agui_thread_id"] = options.ConversationId; + innerOptions.ConversationId = null; + } + + await foreach (var update in base.GetStreamingResponseAsync(messages, innerOptions, cancellationToken).ConfigureAwait(false)) + { + if (conversationId == null && firstUpdate == null) + { + firstUpdate = update; + if (firstUpdate.AdditionalProperties?.TryGetValue("agui_thread_id", out string? threadId) is true) + { + // Capture the thread id from the first update to use as conversation id if none was provided + conversationId = threadId; + } + } + + // Cleanup any temporary approach we used by the handler to avoid issues with FunctionInvokingChatClient + for (var i = 0; i < update.Contents.Count; i++) + { + var content = update.Contents[i]; + if (content is FunctionCallContent functionCallContent) + { + functionCallContent.AdditionalProperties?.Remove("agui_thread_id"); + } + if (content is ServerFunctionCallContent serverFunctionCallContent) + { + update.Contents[i] = serverFunctionCallContent.FunctionCallContent; + } + } + + var finalUpdate = CopyResponseUpdate(update); + + finalUpdate.ConversationId = conversationId; + yield return finalUpdate; + } + } + + private static ChatResponseUpdate CopyResponseUpdate(ChatResponseUpdate source) + { + return new ChatResponseUpdate + { + AuthorName = source.AuthorName, + Role = source.Role, + Contents = source.Contents, + RawRepresentation = source.RawRepresentation, + AdditionalProperties = source.AdditionalProperties, + ResponseId = source.ResponseId, + MessageId = source.MessageId, + CreatedAt = source.CreatedAt, + }; + } + + private sealed class AGUIChatClientHandler : IChatClient + { + private static readonly MediaTypeHeaderValue s_json = new("application/json"); + + private readonly AGUIHttpService _httpService; + private readonly JsonSerializerOptions _jsonSerializerOptions; + private readonly ILogger _logger; + + public AGUIChatClientHandler( + HttpClient httpClient, + string endpoint, + JsonSerializerOptions? jsonSerializerOptions, + IServiceProvider? serviceProvider) + { + this._httpService = new AGUIHttpService(httpClient, endpoint); + this._jsonSerializerOptions = jsonSerializerOptions ?? AGUIJsonSerializerContext.Default.Options; + this._logger = serviceProvider?.GetService(typeof(ILogger)) as ILogger ?? NullLogger.Instance; + + // Use BaseAddress if endpoint is empty, otherwise parse as relative or absolute + Uri metadataUri = string.IsNullOrEmpty(endpoint) && httpClient.BaseAddress is not null + ? httpClient.BaseAddress + : new Uri(endpoint, UriKind.RelativeOrAbsolute); + this.Metadata = new ChatClientMetadata("ag-ui", metadataUri, null); + } + + public ChatClientMetadata Metadata { get; } + + public Task GetResponseAsync( + IEnumerable messages, + ChatOptions? options = null, + CancellationToken cancellationToken = default) + { + return this.GetStreamingResponseAsync(messages, options, cancellationToken) + .ToChatResponseAsync(cancellationToken); + } + + public async IAsyncEnumerable GetStreamingResponseAsync( + IEnumerable messages, + ChatOptions? options = null, + [EnumeratorCancellation] CancellationToken cancellationToken = default) + { + if (messages is null) + { + throw new ArgumentNullException(nameof(messages)); + } + + var runId = $"run_{Guid.NewGuid():N}"; + var messagesList = messages.ToList(); // Avoid triggering the enumerator multiple times. + var threadId = ExtractTemporaryThreadId(messagesList) ?? + ExtractThreadIdFromOptions(options) ?? $"thread_{Guid.NewGuid():N}"; + + // Extract state from the last message if it contains DataContent with application/json + JsonElement state = this.ExtractAndRemoveStateFromMessages(messagesList); + + // Create the input for the AGUI service + var input = new RunAgentInput + { + // AG-UI requires a thread ID to work, but for FunctionInvokingChatClient that + // implies the underlying client is managing the history. + ThreadId = threadId, + RunId = runId, + Messages = messagesList.AsAGUIMessages(this._jsonSerializerOptions), + State = state, + }; + + // Add tools if provided + if (options?.Tools is { Count: > 0 }) + { + input.Tools = options.Tools.AsAGUITools(); + + if (this._logger.IsEnabled(LogLevel.Debug)) + { + this._logger.LogDebug("[AGUIChatClient] Tool count: {ToolCount}", options.Tools.Count); + } + } + + var clientToolSet = new HashSet(); + foreach (var tool in options?.Tools ?? []) + { + clientToolSet.Add(tool.Name); + } + + ChatResponseUpdate? firstUpdate = null; + await foreach (var update in this._httpService.PostRunAsync(input, cancellationToken) + .AsChatResponseUpdatesAsync(this._jsonSerializerOptions, cancellationToken).ConfigureAwait(false)) + { + if (firstUpdate == null) + { + firstUpdate = update; + if (!string.IsNullOrEmpty(firstUpdate.ConversationId) && !string.Equals(firstUpdate.ConversationId, threadId, StringComparison.Ordinal)) + { + threadId = firstUpdate.ConversationId; + } + firstUpdate.AdditionalProperties ??= []; + firstUpdate.AdditionalProperties["agui_thread_id"] = threadId; + } + + if (update.Contents is { Count: 1 } && update.Contents[0] is FunctionCallContent fcc) + { + if (clientToolSet.Contains(fcc.Name)) + { + // Prepare to let the wrapping FunctionInvokingChatClient handle this function call. + // We want to retain the original thread id that either the server sent us or that we set + // in this turn on the next turn, but we can't make it visible to FunctionInvokeingChatClient + // because it would then not send the full history on the next turn as required by AG-UI. + // We store it on additional properties of the function call content, which will be passed down + // in the next turn. + fcc.AdditionalProperties ??= []; + fcc.AdditionalProperties["agui_thread_id"] = threadId; + } + else + { + // Hide the server result call from the FunctionInvokingChatClient. + // The wrapping client will unwrap it and present it as a normal function result. + update.Contents[0] = new ServerFunctionCallContent(fcc); + } + } + + // Remove the conversation id before yielding so that the wrapping FunctionInvokingChatClient + // sends the whole message history on every turn as per AG-UI requirements. + update.ConversationId = null; + yield return update; + } + } + + // Extract the thread id from the options additional properties + private static string? ExtractThreadIdFromOptions(ChatOptions? options) + { + if (options?.AdditionalProperties is null || + !options.AdditionalProperties.TryGetValue("agui_thread_id", out string? threadId) || + string.IsNullOrEmpty(threadId)) + { + return null; + } + return threadId; + } + + // Extract the thread id from the second last message's function call content additional properties + private static string? ExtractTemporaryThreadId(List messagesList) + { + if (messagesList.Count < 2) + { + return null; + } + var functionCall = messagesList[messagesList.Count - 2]; + if (functionCall.Contents.Count < 1 || functionCall.Contents[0] is not FunctionCallContent content) + { + return null; + } + + if (content.AdditionalProperties is null || + !content.AdditionalProperties.TryGetValue("agui_thread_id", out string? threadId) || + string.IsNullOrEmpty(threadId)) + { + return null; + } + + return threadId; + } + + // Extract state from the last message's DataContent with application/json media type + // and remove that message from the list + private JsonElement ExtractAndRemoveStateFromMessages(List messagesList) + { + if (messagesList.Count == 0) + { + return default; + } + + // Check the last message for state DataContent + ChatMessage lastMessage = messagesList[messagesList.Count - 1]; + for (int i = 0; i < lastMessage.Contents.Count; i++) + { + if (lastMessage.Contents[i] is DataContent dataContent && + MediaTypeHeaderValue.TryParse(dataContent.MediaType, out var mediaType) && + mediaType.Equals(s_json)) + { + // Deserialize the state JSON directly from UTF-8 bytes + try + { + JsonElement stateElement = (JsonElement)JsonSerializer.Deserialize( + dataContent.Data.Span, + this._jsonSerializerOptions.GetTypeInfo(typeof(JsonElement)))!; + + // Remove the DataContent from the message contents + lastMessage.Contents.RemoveAt(i); + + // If no contents remain, remove the entire message + if (lastMessage.Contents.Count == 0) + { + messagesList.RemoveAt(messagesList.Count - 1); + } + + return stateElement; + } + catch (JsonException ex) + { + throw new InvalidOperationException($"Failed to deserialize state JSON from DataContent: {ex.Message}", ex); + } + } + } + + return default; + } + + public void Dispose() + { + // No resources to dispose + } + + public object? GetService(Type serviceType, object? serviceKey = null) + { + if (serviceType == typeof(ChatClientMetadata)) + { + return this.Metadata; + } + + return null; + } + } + + private sealed class ServerFunctionCallContent(FunctionCallContent functionCall) : AIContent + { + public FunctionCallContent FunctionCallContent { get; } = functionCall; + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.AGUI/Microsoft.Agents.AI.AGUI.csproj b/dotnet/src/Microsoft.Agents.AI.AGUI/Microsoft.Agents.AI.AGUI.csproj index 8992aaf4fb..57cb375e14 100644 --- a/dotnet/src/Microsoft.Agents.AI.AGUI/Microsoft.Agents.AI.AGUI.csproj +++ b/dotnet/src/Microsoft.Agents.AI.AGUI/Microsoft.Agents.AI.AGUI.csproj @@ -1,18 +1,11 @@ - $(ProjectsTargetFrameworks) - $(ProjectsDebugTargetFrameworks) preview - - - false - - true @@ -28,8 +21,8 @@ - - + + diff --git a/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUIAssistantMessage.cs b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUIAssistantMessage.cs new file mode 100644 index 0000000000..4bf1fdfef4 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUIAssistantMessage.cs @@ -0,0 +1,23 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +#if ASPNETCORE +namespace Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.Shared; +#else +namespace Microsoft.Agents.AI.AGUI.Shared; +#endif + +internal sealed class AGUIAssistantMessage : AGUIMessage +{ + public AGUIAssistantMessage() + { + this.Role = AGUIRoles.Assistant; + } + + [JsonPropertyName("name")] + public string? Name { get; set; } + + [JsonPropertyName("toolCalls")] + public AGUIToolCall[]? ToolCalls { get; set; } +} diff --git a/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUIChatMessageExtensions.cs b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUIChatMessageExtensions.cs index 2b09fb8da2..506956cac8 100644 --- a/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUIChatMessageExtensions.cs +++ b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUIChatMessageExtensions.cs @@ -2,6 +2,7 @@ using System; using System.Collections.Generic; +using System.Text.Json; using Microsoft.Extensions.AI; #if ASPNETCORE @@ -15,28 +16,194 @@ internal static class AGUIChatMessageExtensions private static readonly ChatRole s_developerChatRole = new("developer"); public static IEnumerable AsChatMessages( - this IEnumerable aguiMessages) + this IEnumerable aguiMessages, + JsonSerializerOptions jsonSerializerOptions) { foreach (var message in aguiMessages) { - yield return new ChatMessage( - MapChatRole(message.Role), - message.Content); + var role = MapChatRole(message.Role); + + switch (message) + { + case AGUIToolMessage toolMessage: + { + object? result; + if (string.IsNullOrEmpty(toolMessage.Content)) + { + result = toolMessage.Content; + } + else + { + // Try to deserialize as JSON, but fall back to string if it fails + try + { + result = JsonSerializer.Deserialize(toolMessage.Content, AGUIJsonSerializerContext.Default.JsonElement); + } + catch (JsonException) + { + result = toolMessage.Content; + } + } + + yield return new ChatMessage( + role, + [ + new FunctionResultContent( + toolMessage.ToolCallId, + result) + ]); + break; + } + + case AGUIAssistantMessage assistantMessage when assistantMessage.ToolCalls is { Length: > 0 }: + { + var contents = new List(); + + if (!string.IsNullOrEmpty(assistantMessage.Content)) + { + contents.Add(new TextContent(assistantMessage.Content)); + } + + // Add tool calls + foreach (var toolCall in assistantMessage.ToolCalls) + { + Dictionary? arguments = null; + if (!string.IsNullOrEmpty(toolCall.Function.Arguments)) + { + arguments = (Dictionary?)JsonSerializer.Deserialize( + toolCall.Function.Arguments, + jsonSerializerOptions.GetTypeInfo(typeof(Dictionary))); + } + + contents.Add(new FunctionCallContent( + toolCall.Id, + toolCall.Function.Name, + arguments)); + } + + yield return new ChatMessage(role, contents) + { + MessageId = message.Id + }; + break; + } + + default: + { + string content = message switch + { + AGUIDeveloperMessage dev => dev.Content, + AGUISystemMessage sys => sys.Content, + AGUIUserMessage user => user.Content, + AGUIAssistantMessage asst => asst.Content, + _ => string.Empty + }; + + yield return new ChatMessage(role, content) + { + MessageId = message.Id + }; + break; + } + } } } public static IEnumerable AsAGUIMessages( - this IEnumerable chatMessages) + this IEnumerable chatMessages, + JsonSerializerOptions jsonSerializerOptions) { foreach (var message in chatMessages) { - yield return new AGUIMessage + message.MessageId ??= Guid.NewGuid().ToString("N"); + if (message.Role == ChatRole.Tool) + { + foreach (var toolMessage in MapToolMessages(jsonSerializerOptions, message)) + { + yield return toolMessage; + } + } + else if (message.Role == ChatRole.Assistant) + { + var assistantMessage = MapAssistantMessage(jsonSerializerOptions, message); + if (assistantMessage != null) + { + yield return assistantMessage; + } + } + else + { + yield return message.Role.Value switch + { + AGUIRoles.Developer => new AGUIDeveloperMessage { Id = message.MessageId, Content = message.Text ?? string.Empty }, + AGUIRoles.System => new AGUISystemMessage { Id = message.MessageId, Content = message.Text ?? string.Empty }, + AGUIRoles.User => new AGUIUserMessage { Id = message.MessageId, Content = message.Text ?? string.Empty }, + _ => throw new InvalidOperationException($"Unknown role: {message.Role.Value}") + }; + } + } + } + + private static AGUIAssistantMessage? MapAssistantMessage(JsonSerializerOptions jsonSerializerOptions, ChatMessage message) + { + List? toolCalls = null; + string? textContent = null; + + foreach (var content in message.Contents) + { + if (content is FunctionCallContent functionCall) + { + var argumentsJson = functionCall.Arguments is null ? + "{}" : + JsonSerializer.Serialize(functionCall.Arguments, jsonSerializerOptions.GetTypeInfo(typeof(IDictionary))); + toolCalls ??= []; + toolCalls.Add(new AGUIToolCall + { + Id = functionCall.CallId, + Type = "function", + Function = new AGUIFunctionCall + { + Name = functionCall.Name, + Arguments = argumentsJson + } + }); + } + else if (content is TextContent textContentItem) + { + textContent = textContentItem.Text; + } + } + + // Create message with tool calls and/or text content + if (toolCalls?.Count > 0 || !string.IsNullOrEmpty(textContent)) + { + return new AGUIAssistantMessage { Id = message.MessageId, - Role = message.Role.Value, - Content = message.Text, + Content = textContent ?? string.Empty, + ToolCalls = toolCalls?.Count > 0 ? toolCalls.ToArray() : null }; } + + return null; + } + + private static IEnumerable MapToolMessages(JsonSerializerOptions jsonSerializerOptions, ChatMessage message) + { + foreach (var content in message.Contents) + { + if (content is FunctionResultContent functionResult) + { + yield return new AGUIToolMessage + { + Id = functionResult.CallId, + ToolCallId = functionResult.CallId, + Content = functionResult.Result is null ? + string.Empty : + JsonSerializer.Serialize(functionResult.Result, jsonSerializerOptions.GetTypeInfo(functionResult.Result.GetType())) + }; + } + } } public static ChatRole MapChatRole(string role) => @@ -44,5 +211,6 @@ public static ChatRole MapChatRole(string role) => string.Equals(role, AGUIRoles.User, StringComparison.OrdinalIgnoreCase) ? ChatRole.User : string.Equals(role, AGUIRoles.Assistant, StringComparison.OrdinalIgnoreCase) ? ChatRole.Assistant : string.Equals(role, AGUIRoles.Developer, StringComparison.OrdinalIgnoreCase) ? s_developerChatRole : + string.Equals(role, AGUIRoles.Tool, StringComparison.OrdinalIgnoreCase) ? ChatRole.Tool : throw new InvalidOperationException($"Unknown chat role: {role}"); } diff --git a/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUIContextItem.cs b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUIContextItem.cs new file mode 100644 index 0000000000..54be56f880 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUIContextItem.cs @@ -0,0 +1,18 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +#if ASPNETCORE +namespace Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.Shared; +#else +namespace Microsoft.Agents.AI.AGUI.Shared; +#endif + +internal sealed class AGUIContextItem +{ + [JsonPropertyName("description")] + public string Description { get; set; } = string.Empty; + + [JsonPropertyName("value")] + public string Value { get; set; } = string.Empty; +} diff --git a/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUIDeveloperMessage.cs b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUIDeveloperMessage.cs new file mode 100644 index 0000000000..e41f375b9c --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUIDeveloperMessage.cs @@ -0,0 +1,15 @@ +// Copyright (c) Microsoft. All rights reserved. + +#if ASPNETCORE +namespace Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.Shared; +#else +namespace Microsoft.Agents.AI.AGUI.Shared; +#endif + +internal sealed class AGUIDeveloperMessage : AGUIMessage +{ + public AGUIDeveloperMessage() + { + this.Role = AGUIRoles.Developer; + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUIEventTypes.cs b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUIEventTypes.cs index 74ff3da37f..1b8958cdf0 100644 --- a/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUIEventTypes.cs +++ b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUIEventTypes.cs @@ -19,4 +19,16 @@ internal static class AGUIEventTypes public const string TextMessageContent = "TEXT_MESSAGE_CONTENT"; public const string TextMessageEnd = "TEXT_MESSAGE_END"; + + public const string ToolCallStart = "TOOL_CALL_START"; + + public const string ToolCallArgs = "TOOL_CALL_ARGS"; + + public const string ToolCallEnd = "TOOL_CALL_END"; + + public const string ToolCallResult = "TOOL_CALL_RESULT"; + + public const string StateSnapshot = "STATE_SNAPSHOT"; + + public const string StateDelta = "STATE_DELTA"; } diff --git a/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUIFunctionCall.cs b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUIFunctionCall.cs new file mode 100644 index 0000000000..f69dbcbac6 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUIFunctionCall.cs @@ -0,0 +1,18 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +#if ASPNETCORE +namespace Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.Shared; +#else +namespace Microsoft.Agents.AI.AGUI.Shared; +#endif + +internal sealed class AGUIFunctionCall +{ + [JsonPropertyName("name")] + public string Name { get; set; } = string.Empty; + + [JsonPropertyName("arguments")] + public string Arguments { get; set; } = string.Empty; +} diff --git a/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUIJsonSerializerContext.cs b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUIJsonSerializerContext.cs index fa2e0ced1a..b13a803625 100644 --- a/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUIJsonSerializerContext.cs +++ b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUIJsonSerializerContext.cs @@ -1,5 +1,6 @@ // Copyright (c) Microsoft. All rights reserved. +using System.Collections.Generic; using System.Text.Json.Serialization; #if ASPNETCORE @@ -12,18 +13,50 @@ namespace Microsoft.Agents.AI.Hosting.AGUI.AspNetCore; namespace Microsoft.Agents.AI.AGUI; #endif +// All JsonSerializable attributes below are required for AG-UI functionality: +// - AG-UI message types (AGUIMessage, AGUIUserMessage, etc.) for protocol communication +// - Event types (BaseEvent, RunStartedEvent, etc.) for server-sent events streaming +// - Tool-related types (AGUITool, AGUIToolCall, AGUIFunctionCall) for tool calling support +// - Primitive and dictionary types (string, int, Dictionary, JsonElement) are required for +// serializing tool call parameters and results which can contain arbitrary data types [JsonSourceGenerationOptions(WriteIndented = false, DefaultIgnoreCondition = JsonIgnoreCondition.Never)] [JsonSerializable(typeof(RunAgentInput))] +[JsonSerializable(typeof(AGUIMessage))] +[JsonSerializable(typeof(AGUIMessage[]))] +[JsonSerializable(typeof(AGUIDeveloperMessage))] +[JsonSerializable(typeof(AGUISystemMessage))] +[JsonSerializable(typeof(AGUIUserMessage))] +[JsonSerializable(typeof(AGUIAssistantMessage))] +[JsonSerializable(typeof(AGUIToolMessage))] +[JsonSerializable(typeof(AGUITool))] +[JsonSerializable(typeof(AGUIToolCall))] +[JsonSerializable(typeof(AGUIToolCall[]))] +[JsonSerializable(typeof(AGUIFunctionCall))] [JsonSerializable(typeof(BaseEvent))] +[JsonSerializable(typeof(BaseEvent[]))] [JsonSerializable(typeof(RunStartedEvent))] [JsonSerializable(typeof(RunFinishedEvent))] [JsonSerializable(typeof(RunErrorEvent))] [JsonSerializable(typeof(TextMessageStartEvent))] [JsonSerializable(typeof(TextMessageContentEvent))] [JsonSerializable(typeof(TextMessageEndEvent))] -#if !ASPNETCORE -[JsonSerializable(typeof(AGUIAgentThread.AGUIAgentThreadState))] -#endif -internal partial class AGUIJsonSerializerContext : JsonSerializerContext -{ -} +[JsonSerializable(typeof(ToolCallStartEvent))] +[JsonSerializable(typeof(ToolCallArgsEvent))] +[JsonSerializable(typeof(ToolCallEndEvent))] +[JsonSerializable(typeof(ToolCallResultEvent))] +[JsonSerializable(typeof(StateSnapshotEvent))] +[JsonSerializable(typeof(StateDeltaEvent))] +[JsonSerializable(typeof(IDictionary))] +[JsonSerializable(typeof(Dictionary))] +[JsonSerializable(typeof(IDictionary))] +[JsonSerializable(typeof(Dictionary))] +[JsonSerializable(typeof(System.Text.Json.JsonElement))] +[JsonSerializable(typeof(Dictionary))] +[JsonSerializable(typeof(string))] +[JsonSerializable(typeof(int))] +[JsonSerializable(typeof(long))] +[JsonSerializable(typeof(double))] +[JsonSerializable(typeof(float))] +[JsonSerializable(typeof(bool))] +[JsonSerializable(typeof(decimal))] +internal sealed partial class AGUIJsonSerializerContext : JsonSerializerContext; diff --git a/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUIMessage.cs b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUIMessage.cs index b32c1efcfa..01ccb07b15 100644 --- a/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUIMessage.cs +++ b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUIMessage.cs @@ -8,7 +8,8 @@ namespace Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.Shared; namespace Microsoft.Agents.AI.AGUI.Shared; #endif -internal sealed class AGUIMessage +[JsonConverter(typeof(AGUIMessageJsonConverter))] +internal abstract class AGUIMessage { [JsonPropertyName("id")] public string? Id { get; set; } diff --git a/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUIMessageJsonConverter.cs b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUIMessageJsonConverter.cs new file mode 100644 index 0000000000..ceb0504c63 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUIMessageJsonConverter.cs @@ -0,0 +1,82 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Text.Json; +using System.Text.Json.Serialization; + +#if ASPNETCORE +namespace Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.Shared; +#else +namespace Microsoft.Agents.AI.AGUI.Shared; +#endif + +internal sealed class AGUIMessageJsonConverter : JsonConverter +{ + private const string RoleDiscriminatorPropertyName = "role"; + + public override bool CanConvert(Type typeToConvert) => + typeof(AGUIMessage).IsAssignableFrom(typeToConvert); + + public override AGUIMessage Read( + ref Utf8JsonReader reader, + Type typeToConvert, + JsonSerializerOptions options) + { + var jsonElementTypeInfo = options.GetTypeInfo(typeof(JsonElement)); + JsonElement jsonElement = (JsonElement)JsonSerializer.Deserialize(ref reader, jsonElementTypeInfo)!; + + // Try to get the discriminator property + if (!jsonElement.TryGetProperty(RoleDiscriminatorPropertyName, out JsonElement discriminatorElement)) + { + throw new JsonException($"Missing required property '{RoleDiscriminatorPropertyName}' for AGUIMessage deserialization"); + } + + string? discriminator = discriminatorElement.GetString(); + + // Map discriminator to concrete type and deserialize using type info from options + AGUIMessage? result = discriminator switch + { + AGUIRoles.Developer => jsonElement.Deserialize(options.GetTypeInfo(typeof(AGUIDeveloperMessage))) as AGUIDeveloperMessage, + AGUIRoles.System => jsonElement.Deserialize(options.GetTypeInfo(typeof(AGUISystemMessage))) as AGUISystemMessage, + AGUIRoles.User => jsonElement.Deserialize(options.GetTypeInfo(typeof(AGUIUserMessage))) as AGUIUserMessage, + AGUIRoles.Assistant => jsonElement.Deserialize(options.GetTypeInfo(typeof(AGUIAssistantMessage))) as AGUIAssistantMessage, + AGUIRoles.Tool => jsonElement.Deserialize(options.GetTypeInfo(typeof(AGUIToolMessage))) as AGUIToolMessage, + _ => throw new JsonException($"Unknown AGUIMessage role discriminator: '{discriminator}'") + }; + + if (result == null) + { + throw new JsonException($"Failed to deserialize AGUIMessage with role discriminator: '{discriminator}'"); + } + + return result; + } + + public override void Write( + Utf8JsonWriter writer, + AGUIMessage value, + JsonSerializerOptions options) + { + // Serialize the concrete type directly using type info from options + switch (value) + { + case AGUIDeveloperMessage developer: + JsonSerializer.Serialize(writer, developer, options.GetTypeInfo(typeof(AGUIDeveloperMessage))); + break; + case AGUISystemMessage system: + JsonSerializer.Serialize(writer, system, options.GetTypeInfo(typeof(AGUISystemMessage))); + break; + case AGUIUserMessage user: + JsonSerializer.Serialize(writer, user, options.GetTypeInfo(typeof(AGUIUserMessage))); + break; + case AGUIAssistantMessage assistant: + JsonSerializer.Serialize(writer, assistant, options.GetTypeInfo(typeof(AGUIAssistantMessage))); + break; + case AGUIToolMessage tool: + JsonSerializer.Serialize(writer, tool, options.GetTypeInfo(typeof(AGUIToolMessage))); + break; + default: + throw new JsonException($"Unknown AGUIMessage type: {value.GetType().Name}"); + } + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUIRoles.cs b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUIRoles.cs index fe67224efe..f702d5ec8d 100644 --- a/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUIRoles.cs +++ b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUIRoles.cs @@ -15,4 +15,6 @@ internal static class AGUIRoles public const string Assistant = "assistant"; public const string Developer = "developer"; + + public const string Tool = "tool"; } diff --git a/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUISystemMessage.cs b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUISystemMessage.cs new file mode 100644 index 0000000000..f2d053c23e --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUISystemMessage.cs @@ -0,0 +1,15 @@ +// Copyright (c) Microsoft. All rights reserved. + +#if ASPNETCORE +namespace Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.Shared; +#else +namespace Microsoft.Agents.AI.AGUI.Shared; +#endif + +internal sealed class AGUISystemMessage : AGUIMessage +{ + public AGUISystemMessage() + { + this.Role = AGUIRoles.System; + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUITool.cs b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUITool.cs new file mode 100644 index 0000000000..c42556dcb0 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUITool.cs @@ -0,0 +1,22 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json; +using System.Text.Json.Serialization; + +#if ASPNETCORE +namespace Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.Shared; +#else +namespace Microsoft.Agents.AI.AGUI.Shared; +#endif + +internal sealed class AGUITool +{ + [JsonPropertyName("name")] + public string Name { get; set; } = string.Empty; + + [JsonPropertyName("description")] + public string? Description { get; set; } + + [JsonPropertyName("parameters")] + public JsonElement Parameters { get; set; } +} diff --git a/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUIToolCall.cs b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUIToolCall.cs new file mode 100644 index 0000000000..ca28d956d3 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUIToolCall.cs @@ -0,0 +1,21 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +#if ASPNETCORE +namespace Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.Shared; +#else +namespace Microsoft.Agents.AI.AGUI.Shared; +#endif + +internal sealed class AGUIToolCall +{ + [JsonPropertyName("id")] + public string Id { get; set; } = string.Empty; + + [JsonPropertyName("type")] + public string Type { get; set; } = "function"; + + [JsonPropertyName("function")] + public AGUIFunctionCall Function { get; set; } = new(); +} diff --git a/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUIToolMessage.cs b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUIToolMessage.cs new file mode 100644 index 0000000000..bcd49d2b6f --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUIToolMessage.cs @@ -0,0 +1,23 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +#if ASPNETCORE +namespace Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.Shared; +#else +namespace Microsoft.Agents.AI.AGUI.Shared; +#endif + +internal sealed class AGUIToolMessage : AGUIMessage +{ + public AGUIToolMessage() + { + this.Role = AGUIRoles.Tool; + } + + [JsonPropertyName("toolCallId")] + public string ToolCallId { get; set; } = string.Empty; + + [JsonPropertyName("error")] + public string? Error { get; set; } +} diff --git a/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUIUserMessage.cs b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUIUserMessage.cs new file mode 100644 index 0000000000..e8e9f2ed57 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AGUIUserMessage.cs @@ -0,0 +1,20 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +#if ASPNETCORE +namespace Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.Shared; +#else +namespace Microsoft.Agents.AI.AGUI.Shared; +#endif + +internal sealed class AGUIUserMessage : AGUIMessage +{ + public AGUIUserMessage() + { + this.Role = AGUIRoles.User; + } + + [JsonPropertyName("name")] + public string? Name { get; set; } +} diff --git a/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AIToolExtensions.cs b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AIToolExtensions.cs new file mode 100644 index 0000000000..8952f38a28 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AIToolExtensions.cs @@ -0,0 +1,56 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Collections.Generic; +using Microsoft.Extensions.AI; + +#if ASPNETCORE +namespace Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.Shared; +#else +namespace Microsoft.Agents.AI.AGUI.Shared; +#endif + +internal static class AIToolExtensions +{ + public static IEnumerable AsAGUITools(this IEnumerable tools) + { + if (tools is null) + { + yield break; + } + + foreach (var tool in tools) + { + // Convert both AIFunctionDeclaration and AIFunction (which extends it) to AGUITool + // For AIFunction, we send only the metadata (Name, Description, JsonSchema) + // The actual executable implementation stays on the client side + if (tool is AIFunctionDeclaration function) + { + yield return new AGUITool + { + Name = function.Name, + Description = function.Description, + Parameters = function.JsonSchema + }; + } + } + } + + public static IEnumerable AsAITools(this IEnumerable tools) + { + if (tools is null) + { + yield break; + } + + foreach (var tool in tools) + { + // Create a function declaration from the AG-UI tool definition + // Note: These are declaration-only and cannot be invoked, as the actual + // implementation exists on the client side + yield return AIFunctionFactory.CreateDeclaration( + name: tool.Name, + description: tool.Description, + jsonSchema: tool.Parameters); + } + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AgentRunResponseUpdateAGUIExtensions.cs b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AgentRunResponseUpdateAGUIExtensions.cs deleted file mode 100644 index 59755d7b5a..0000000000 --- a/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/AgentRunResponseUpdateAGUIExtensions.cs +++ /dev/null @@ -1,161 +0,0 @@ -// Copyright (c) Microsoft. All rights reserved. - -using System; -using System.Collections.Generic; -using System.Runtime.CompilerServices; -using System.Threading; -using System.Threading.Tasks; -using Microsoft.Extensions.AI; - -#if ASPNETCORE -namespace Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.Shared; -#else -namespace Microsoft.Agents.AI.AGUI.Shared; -#endif - -internal static class AgentRunResponseUpdateAGUIExtensions -{ -#if !ASPNETCORE - public static async IAsyncEnumerable AsAgentRunResponseUpdatesAsync( - this IAsyncEnumerable events, - [EnumeratorCancellation] CancellationToken cancellationToken = default) - { - string? currentMessageId = null; - ChatRole currentRole = default!; - string? conversationId = null; - string? responseId = null; - await foreach (var evt in events.WithCancellation(cancellationToken).ConfigureAwait(false)) - { - switch (evt) - { - case RunStartedEvent runStarted: - conversationId = runStarted.ThreadId; - responseId = runStarted.RunId; - yield return new AgentRunResponseUpdate(new ChatResponseUpdate( - ChatRole.Assistant, - []) - { - ConversationId = conversationId, - ResponseId = responseId, - CreatedAt = DateTimeOffset.UtcNow - }); - break; - case RunFinishedEvent runFinished: - if (!string.Equals(runFinished.ThreadId, conversationId, StringComparison.Ordinal)) - { - throw new InvalidOperationException($"The run finished event didn't match the run started event thread ID: {runFinished.ThreadId}, {conversationId}"); - } - if (!string.Equals(runFinished.RunId, responseId, StringComparison.Ordinal)) - { - throw new InvalidOperationException($"The run finished event didn't match the run started event run ID: {runFinished.RunId}, {responseId}"); - } - yield return new AgentRunResponseUpdate(new ChatResponseUpdate( - ChatRole.Assistant, runFinished.Result?.GetRawText()) - { - ConversationId = conversationId, - ResponseId = responseId, - CreatedAt = DateTimeOffset.UtcNow - }); - break; - case RunErrorEvent runError: - yield return new AgentRunResponseUpdate(new ChatResponseUpdate( - ChatRole.Assistant, - [(new ErrorContent(runError.Message) { ErrorCode = runError.Code })])); - break; - case TextMessageStartEvent textStart: - if (currentRole != default || currentMessageId != null) - { - throw new InvalidOperationException("Received TextMessageStartEvent while another message is being processed."); - } - - currentRole = AGUIChatMessageExtensions.MapChatRole(textStart.Role); - currentMessageId = textStart.MessageId; - break; - case TextMessageContentEvent textContent: - yield return new AgentRunResponseUpdate(new ChatResponseUpdate( - currentRole, - textContent.Delta) - { - ConversationId = conversationId, - ResponseId = responseId, - MessageId = textContent.MessageId, - CreatedAt = DateTimeOffset.UtcNow - }); - break; - case TextMessageEndEvent textEnd: - if (currentMessageId != textEnd.MessageId) - { - throw new InvalidOperationException("Received TextMessageEndEvent for a different message than the current one."); - } - currentRole = default!; - currentMessageId = null; - break; - } - } - } -#endif - - public static async IAsyncEnumerable AsAGUIEventStreamAsync( - this IAsyncEnumerable updates, - string threadId, - string runId, - [EnumeratorCancellation] CancellationToken cancellationToken = default) - { - yield return new RunStartedEvent - { - ThreadId = threadId, - RunId = runId - }; - - string? currentMessageId = null; - await foreach (var update in updates.WithCancellation(cancellationToken).ConfigureAwait(false)) - { - var chatResponse = update.AsChatResponseUpdate(); - if (chatResponse is { Contents.Count: > 0 } && chatResponse.Contents[0] is TextContent && !string.Equals(currentMessageId, chatResponse.MessageId, StringComparison.Ordinal)) - { - // End the previous message if there was one - if (currentMessageId is not null) - { - yield return new TextMessageEndEvent - { - MessageId = currentMessageId - }; - } - - // Start the new message - yield return new TextMessageStartEvent - { - MessageId = chatResponse.MessageId!, - Role = chatResponse.Role!.Value.Value - }; - - currentMessageId = chatResponse.MessageId; - } - - // Emit text content if present - if (chatResponse is { Contents.Count: > 0 } && chatResponse.Contents[0] is TextContent textContent) - { - yield return new TextMessageContentEvent - { - MessageId = chatResponse.MessageId!, - Delta = textContent.Text ?? string.Empty - }; - } - } - - // End the last message if there was one - if (currentMessageId is not null) - { - yield return new TextMessageEndEvent - { - MessageId = currentMessageId - }; - } - - yield return new RunFinishedEvent - { - ThreadId = threadId, - RunId = runId, - }; - } -} diff --git a/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/BaseEventJsonConverter.cs b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/BaseEventJsonConverter.cs index 58624ac45c..eca2131f23 100644 --- a/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/BaseEventJsonConverter.cs +++ b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/BaseEventJsonConverter.cs @@ -10,10 +10,6 @@ namespace Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.Shared; namespace Microsoft.Agents.AI.AGUI.Shared; #endif -/// -/// Custom JSON converter for polymorphic deserialization of BaseEvent and its derived types. -/// Uses the "type" property as a discriminator to determine the concrete type to deserialize. -/// internal sealed class BaseEventJsonConverter : JsonConverter { private const string TypeDiscriminatorPropertyName = "type"; @@ -26,9 +22,8 @@ public override BaseEvent Read( Type typeToConvert, JsonSerializerOptions options) { - // Parse the JSON into a JsonDocument to inspect properties - using JsonDocument document = JsonDocument.ParseValue(ref reader); - JsonElement jsonElement = document.RootElement.Clone(); + var jsonElementTypeInfo = options.GetTypeInfo(typeof(JsonElement)); + JsonElement jsonElement = (JsonElement)JsonSerializer.Deserialize(ref reader, jsonElementTypeInfo)!; // Try to get the discriminator property if (!jsonElement.TryGetProperty(TypeDiscriminatorPropertyName, out JsonElement discriminatorElement)) @@ -38,21 +33,20 @@ public override BaseEvent Read( string? discriminator = discriminatorElement.GetString(); -#if ASPNETCORE - AGUIJsonSerializerContext context = (AGUIJsonSerializerContext)options.TypeInfoResolver!; -#else - AGUIJsonSerializerContext context = AGUIJsonSerializerContext.Default; -#endif - - // Map discriminator to concrete type and deserialize using the serializer context + // Map discriminator to concrete type and deserialize using type info from options BaseEvent? result = discriminator switch { - AGUIEventTypes.RunStarted => jsonElement.Deserialize(context.RunStartedEvent), - AGUIEventTypes.RunFinished => jsonElement.Deserialize(context.RunFinishedEvent), - AGUIEventTypes.RunError => jsonElement.Deserialize(context.RunErrorEvent), - AGUIEventTypes.TextMessageStart => jsonElement.Deserialize(context.TextMessageStartEvent), - AGUIEventTypes.TextMessageContent => jsonElement.Deserialize(context.TextMessageContentEvent), - AGUIEventTypes.TextMessageEnd => jsonElement.Deserialize(context.TextMessageEndEvent), + AGUIEventTypes.RunStarted => jsonElement.Deserialize(options.GetTypeInfo(typeof(RunStartedEvent))) as RunStartedEvent, + AGUIEventTypes.RunFinished => jsonElement.Deserialize(options.GetTypeInfo(typeof(RunFinishedEvent))) as RunFinishedEvent, + AGUIEventTypes.RunError => jsonElement.Deserialize(options.GetTypeInfo(typeof(RunErrorEvent))) as RunErrorEvent, + AGUIEventTypes.TextMessageStart => jsonElement.Deserialize(options.GetTypeInfo(typeof(TextMessageStartEvent))) as TextMessageStartEvent, + AGUIEventTypes.TextMessageContent => jsonElement.Deserialize(options.GetTypeInfo(typeof(TextMessageContentEvent))) as TextMessageContentEvent, + AGUIEventTypes.TextMessageEnd => jsonElement.Deserialize(options.GetTypeInfo(typeof(TextMessageEndEvent))) as TextMessageEndEvent, + AGUIEventTypes.ToolCallStart => jsonElement.Deserialize(options.GetTypeInfo(typeof(ToolCallStartEvent))) as ToolCallStartEvent, + AGUIEventTypes.ToolCallArgs => jsonElement.Deserialize(options.GetTypeInfo(typeof(ToolCallArgsEvent))) as ToolCallArgsEvent, + AGUIEventTypes.ToolCallEnd => jsonElement.Deserialize(options.GetTypeInfo(typeof(ToolCallEndEvent))) as ToolCallEndEvent, + AGUIEventTypes.ToolCallResult => jsonElement.Deserialize(options.GetTypeInfo(typeof(ToolCallResultEvent))) as ToolCallResultEvent, + AGUIEventTypes.StateSnapshot => jsonElement.Deserialize(options.GetTypeInfo(typeof(StateSnapshotEvent))) as StateSnapshotEvent, _ => throw new JsonException($"Unknown BaseEvent type discriminator: '{discriminator}'") }; @@ -69,35 +63,47 @@ public override void Write( BaseEvent value, JsonSerializerOptions options) { -#if ASPNETCORE - AGUIJsonSerializerContext context = (AGUIJsonSerializerContext)options.TypeInfoResolver!; -#else - AGUIJsonSerializerContext context = AGUIJsonSerializerContext.Default; -#endif - - // Serialize the concrete type directly using the serializer context + // Serialize the concrete type directly using type info from options switch (value) { case RunStartedEvent runStarted: - JsonSerializer.Serialize(writer, runStarted, context.RunStartedEvent); + JsonSerializer.Serialize(writer, runStarted, options.GetTypeInfo(typeof(RunStartedEvent))); break; case RunFinishedEvent runFinished: - JsonSerializer.Serialize(writer, runFinished, context.RunFinishedEvent); + JsonSerializer.Serialize(writer, runFinished, options.GetTypeInfo(typeof(RunFinishedEvent))); break; case RunErrorEvent runError: - JsonSerializer.Serialize(writer, runError, context.RunErrorEvent); + JsonSerializer.Serialize(writer, runError, options.GetTypeInfo(typeof(RunErrorEvent))); break; case TextMessageStartEvent textStart: - JsonSerializer.Serialize(writer, textStart, context.TextMessageStartEvent); + JsonSerializer.Serialize(writer, textStart, options.GetTypeInfo(typeof(TextMessageStartEvent))); break; case TextMessageContentEvent textContent: - JsonSerializer.Serialize(writer, textContent, context.TextMessageContentEvent); + JsonSerializer.Serialize(writer, textContent, options.GetTypeInfo(typeof(TextMessageContentEvent))); break; case TextMessageEndEvent textEnd: - JsonSerializer.Serialize(writer, textEnd, context.TextMessageEndEvent); + JsonSerializer.Serialize(writer, textEnd, options.GetTypeInfo(typeof(TextMessageEndEvent))); + break; + case ToolCallStartEvent toolCallStart: + JsonSerializer.Serialize(writer, toolCallStart, options.GetTypeInfo(typeof(ToolCallStartEvent))); + break; + case ToolCallArgsEvent toolCallArgs: + JsonSerializer.Serialize(writer, toolCallArgs, options.GetTypeInfo(typeof(ToolCallArgsEvent))); + break; + case ToolCallEndEvent toolCallEnd: + JsonSerializer.Serialize(writer, toolCallEnd, options.GetTypeInfo(typeof(ToolCallEndEvent))); + break; + case ToolCallResultEvent toolCallResult: + JsonSerializer.Serialize(writer, toolCallResult, options.GetTypeInfo(typeof(ToolCallResultEvent))); + break; + case StateSnapshotEvent stateSnapshot: + JsonSerializer.Serialize(writer, stateSnapshot, options.GetTypeInfo(typeof(StateSnapshotEvent))); + break; + case StateDeltaEvent stateDelta: + JsonSerializer.Serialize(writer, stateDelta, options.GetTypeInfo(typeof(StateDeltaEvent))); break; default: - throw new JsonException($"Unknown BaseEvent type: {value.GetType().Name}"); + throw new InvalidOperationException($"Unknown event type: {value.GetType().Name}"); } } } diff --git a/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/ChatResponseUpdateAGUIExtensions.cs b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/ChatResponseUpdateAGUIExtensions.cs new file mode 100644 index 0000000000..f5fb103bd4 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/ChatResponseUpdateAGUIExtensions.cs @@ -0,0 +1,496 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Collections.Generic; +using System.Diagnostics; +using System.Net.Http.Headers; +using System.Runtime.CompilerServices; +using System.Text; +using System.Text.Json; +using System.Threading; +using System.Threading.Tasks; +using Microsoft.Extensions.AI; + +#if ASPNETCORE +namespace Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.Shared; +#else +namespace Microsoft.Agents.AI.AGUI.Shared; +#endif + +internal static class ChatResponseUpdateAGUIExtensions +{ + private static readonly MediaTypeHeaderValue? s_jsonPatchMediaType = new("application/json-patch+json"); + private static readonly MediaTypeHeaderValue? s_json = new("application/json"); + + public static async IAsyncEnumerable AsChatResponseUpdatesAsync( + this IAsyncEnumerable events, + JsonSerializerOptions jsonSerializerOptions, + [EnumeratorCancellation] CancellationToken cancellationToken = default) + { + string? conversationId = null; + string? responseId = null; + var textMessageBuilder = new TextMessageBuilder(); + var toolCallAccumulator = new ToolCallBuilder(); + await foreach (var evt in events.WithCancellation(cancellationToken).ConfigureAwait(false)) + { + switch (evt) + { + // Lifecycle events + case RunStartedEvent runStarted: + conversationId = runStarted.ThreadId; + responseId = runStarted.RunId; + toolCallAccumulator.SetConversationAndResponseIds(conversationId, responseId); + textMessageBuilder.SetConversationAndResponseIds(conversationId, responseId); + yield return ValidateAndEmitRunStart(runStarted); + break; + case RunFinishedEvent runFinished: + yield return ValidateAndEmitRunFinished(conversationId, responseId, runFinished); + break; + case RunErrorEvent runError: + yield return new ChatResponseUpdate(ChatRole.Assistant, [(new ErrorContent(runError.Message) { ErrorCode = runError.Code })]); + break; + + // Text events + case TextMessageStartEvent textStart: + textMessageBuilder.AddTextStart(textStart); + break; + case TextMessageContentEvent textContent: + yield return textMessageBuilder.EmitTextUpdate(textContent); + break; + case TextMessageEndEvent textEnd: + textMessageBuilder.EndCurrentMessage(textEnd); + break; + + // Tool call events + case ToolCallStartEvent toolCallStart: + toolCallAccumulator.AddToolCallStart(toolCallStart); + break; + case ToolCallArgsEvent toolCallArgs: + toolCallAccumulator.AddToolCallArgs(toolCallArgs, jsonSerializerOptions); + break; + case ToolCallEndEvent toolCallEnd: + yield return toolCallAccumulator.EmitToolCallUpdate(toolCallEnd, jsonSerializerOptions); + break; + case ToolCallResultEvent toolCallResult: + yield return toolCallAccumulator.EmitToolCallResult(toolCallResult, jsonSerializerOptions); + break; + + // State snapshot events + case StateSnapshotEvent stateSnapshot: + if (stateSnapshot.Snapshot.HasValue) + { + yield return CreateStateSnapshotUpdate(stateSnapshot, conversationId, responseId, jsonSerializerOptions); + } + break; + case StateDeltaEvent stateDelta: + if (stateDelta.Delta.HasValue) + { + yield return CreateStateDeltaUpdate(stateDelta, conversationId, responseId, jsonSerializerOptions); + } + break; + } + } + } + + private static ChatResponseUpdate CreateStateSnapshotUpdate( + StateSnapshotEvent stateSnapshot, + string? conversationId, + string? responseId, + JsonSerializerOptions jsonSerializerOptions) + { + // Serialize JsonElement directly to UTF-8 bytes using AOT-safe overload + byte[] jsonBytes = JsonSerializer.SerializeToUtf8Bytes( + stateSnapshot.Snapshot!.Value, + jsonSerializerOptions.GetTypeInfo(typeof(JsonElement))); + DataContent dataContent = new(jsonBytes, "application/json"); + + return new ChatResponseUpdate(ChatRole.Assistant, [dataContent]) + { + ConversationId = conversationId, + ResponseId = responseId, + CreatedAt = DateTimeOffset.UtcNow, + AdditionalProperties = new AdditionalPropertiesDictionary + { + ["is_state_snapshot"] = true + } + }; + } + + private static ChatResponseUpdate CreateStateDeltaUpdate( + StateDeltaEvent stateDelta, + string? conversationId, + string? responseId, + JsonSerializerOptions jsonSerializerOptions) + { + // Serialize JsonElement directly to UTF-8 bytes using AOT-safe overload + byte[] jsonBytes = JsonSerializer.SerializeToUtf8Bytes( + stateDelta.Delta!.Value, + jsonSerializerOptions.GetTypeInfo(typeof(JsonElement))); + DataContent dataContent = new(jsonBytes, "application/json-patch+json"); + + return new ChatResponseUpdate(ChatRole.Assistant, [dataContent]) + { + ConversationId = conversationId, + ResponseId = responseId, + CreatedAt = DateTimeOffset.UtcNow, + AdditionalProperties = new AdditionalPropertiesDictionary + { + ["is_state_delta"] = true + } + }; + } + + private sealed class TextMessageBuilder() + { + private ChatRole _currentRole; + private string? _currentMessageId; + private string? _conversationId; + private string? _responseId; + + public void SetConversationAndResponseIds(string? conversationId, string? responseId) + { + this._conversationId = conversationId; + this._responseId = responseId; + } + + public void AddTextStart(TextMessageStartEvent textStart) + { + if (this._currentRole != default || this._currentMessageId != null) + { + throw new InvalidOperationException("Received TextMessageStartEvent while another message is being processed."); + } + + this._currentRole = AGUIChatMessageExtensions.MapChatRole(textStart.Role); + this._currentMessageId = textStart.MessageId; + } + + internal ChatResponseUpdate EmitTextUpdate(TextMessageContentEvent textContent) + { + return new ChatResponseUpdate( + this._currentRole, + textContent.Delta) + { + ConversationId = this._conversationId, + ResponseId = this._responseId, + MessageId = textContent.MessageId, + CreatedAt = DateTimeOffset.UtcNow + }; + } + + internal void EndCurrentMessage(TextMessageEndEvent textEnd) + { + if (this._currentMessageId != textEnd.MessageId) + { + throw new InvalidOperationException("Received TextMessageEndEvent for a different message than the current one."); + } + this._currentRole = default; + this._currentMessageId = null; + } + } + + private static ChatResponseUpdate ValidateAndEmitRunStart(RunStartedEvent runStarted) + { + return new ChatResponseUpdate( + ChatRole.Assistant, + []) + { + ConversationId = runStarted.ThreadId, + ResponseId = runStarted.RunId, + CreatedAt = DateTimeOffset.UtcNow + }; + } + + private static ChatResponseUpdate ValidateAndEmitRunFinished(string? conversationId, string? responseId, RunFinishedEvent runFinished) + { + if (!string.Equals(runFinished.ThreadId, conversationId, StringComparison.Ordinal)) + { + throw new InvalidOperationException($"The run finished event didn't match the run started event thread ID: {runFinished.ThreadId}, {conversationId}"); + } + if (!string.Equals(runFinished.RunId, responseId, StringComparison.Ordinal)) + { + throw new InvalidOperationException($"The run finished event didn't match the run started event run ID: {runFinished.RunId}, {responseId}"); + } + + return new ChatResponseUpdate( + ChatRole.Assistant, runFinished.Result?.GetRawText()) + { + ConversationId = conversationId, + ResponseId = responseId, + CreatedAt = DateTimeOffset.UtcNow + }; + } + + private sealed class ToolCallBuilder + { + private string? _conversationId; + private string? _responseId; + private StringBuilder? _accumulatedArgs; + private FunctionCallContent? _currentFunctionCall; + + public void AddToolCallStart(ToolCallStartEvent toolCallStart) + { + if (this._currentFunctionCall != null) + { + throw new InvalidOperationException("Received ToolCallStartEvent while another tool call is being processed."); + } + this._accumulatedArgs ??= new StringBuilder(); + this._currentFunctionCall = new( + toolCallStart.ToolCallId, + toolCallStart.ToolCallName, + null); + } + + public void AddToolCallArgs(ToolCallArgsEvent toolCallArgs, JsonSerializerOptions options) + { + if (this._currentFunctionCall == null) + { + throw new InvalidOperationException("Received ToolCallArgsEvent without a current tool call."); + } + + if (!string.Equals(this._currentFunctionCall.CallId, toolCallArgs.ToolCallId, StringComparison.Ordinal)) + { + throw new InvalidOperationException("Received ToolCallArgsEvent for a different tool call than the current one."); + } + + Debug.Assert(this._accumulatedArgs != null, "Accumulated args should have been initialized in ToolCallStartEvent."); + this._accumulatedArgs.Append(toolCallArgs.Delta); + } + + internal ChatResponseUpdate EmitToolCallUpdate(ToolCallEndEvent toolCallEnd, JsonSerializerOptions jsonSerializerOptions) + { + if (this._currentFunctionCall == null) + { + throw new InvalidOperationException("Received ToolCallEndEvent without a current tool call."); + } + if (!string.Equals(this._currentFunctionCall.CallId, toolCallEnd.ToolCallId, StringComparison.Ordinal)) + { + throw new InvalidOperationException("Received ToolCallEndEvent for a different tool call than the current one."); + } + Debug.Assert(this._accumulatedArgs != null, "Accumulated args should have been initialized in ToolCallStartEvent."); + var arguments = DeserializeArgumentsIfAvailable(this._accumulatedArgs.ToString(), jsonSerializerOptions); + this._accumulatedArgs.Clear(); + this._currentFunctionCall.Arguments = arguments; + var invocation = this._currentFunctionCall; + this._currentFunctionCall = null; + return new ChatResponseUpdate( + ChatRole.Assistant, + [invocation]) + { + ConversationId = this._conversationId, + ResponseId = this._responseId, + MessageId = invocation.CallId, + CreatedAt = DateTimeOffset.UtcNow + }; + } + + public ChatResponseUpdate EmitToolCallResult(ToolCallResultEvent toolCallResult, JsonSerializerOptions options) + { + return new ChatResponseUpdate( + ChatRole.Tool, + [new FunctionResultContent( + toolCallResult.ToolCallId, + DeserializeResultIfAvailable(toolCallResult, options))]) + { + ConversationId = this._conversationId, + ResponseId = this._responseId, + MessageId = toolCallResult.MessageId, + CreatedAt = DateTimeOffset.UtcNow + }; + } + + internal void SetConversationAndResponseIds(string conversationId, string responseId) + { + this._conversationId = conversationId; + this._responseId = responseId; + } + } + + private static IDictionary? DeserializeArgumentsIfAvailable(string argsJson, JsonSerializerOptions options) + { + if (!string.IsNullOrEmpty(argsJson)) + { + return (IDictionary?)JsonSerializer.Deserialize( + argsJson, + options.GetTypeInfo(typeof(IDictionary))); + } + + return null; + } + + private static object? DeserializeResultIfAvailable(ToolCallResultEvent toolCallResult, JsonSerializerOptions options) + { + if (!string.IsNullOrEmpty(toolCallResult.Content)) + { + return JsonSerializer.Deserialize(toolCallResult.Content, options.GetTypeInfo(typeof(JsonElement))); + } + + return null; + } + + public static async IAsyncEnumerable AsAGUIEventStreamAsync( + this IAsyncEnumerable updates, + string threadId, + string runId, + JsonSerializerOptions jsonSerializerOptions, + [EnumeratorCancellation] CancellationToken cancellationToken = default) + { + yield return new RunStartedEvent + { + ThreadId = threadId, + RunId = runId + }; + + string? currentMessageId = null; + await foreach (var chatResponse in updates.WithCancellation(cancellationToken).ConfigureAwait(false)) + { + if (chatResponse is { Contents.Count: > 0 } && + chatResponse.Contents[0] is TextContent && + !string.Equals(currentMessageId, chatResponse.MessageId, StringComparison.Ordinal)) + { + // End the previous message if there was one + if (currentMessageId is not null) + { + yield return new TextMessageEndEvent + { + MessageId = currentMessageId + }; + } + + // Start the new message + yield return new TextMessageStartEvent + { + MessageId = chatResponse.MessageId!, + Role = chatResponse.Role!.Value.Value + }; + + currentMessageId = chatResponse.MessageId; + } + + // Emit text content if present + if (chatResponse is { Contents.Count: > 0 } && chatResponse.Contents[0] is TextContent textContent && + !string.IsNullOrEmpty(textContent.Text)) + { + yield return new TextMessageContentEvent + { + MessageId = chatResponse.MessageId!, + Delta = textContent.Text + }; + } + + // Emit tool call events and tool result events + if (chatResponse is { Contents.Count: > 0 }) + { + foreach (var content in chatResponse.Contents) + { + if (content is FunctionCallContent functionCallContent) + { + yield return new ToolCallStartEvent + { + ToolCallId = functionCallContent.CallId, + ToolCallName = functionCallContent.Name, + ParentMessageId = chatResponse.MessageId + }; + + yield return new ToolCallArgsEvent + { + ToolCallId = functionCallContent.CallId, + Delta = JsonSerializer.Serialize( + functionCallContent.Arguments, + jsonSerializerOptions.GetTypeInfo(typeof(IDictionary))) + }; + + yield return new ToolCallEndEvent + { + ToolCallId = functionCallContent.CallId + }; + } + else if (content is FunctionResultContent functionResultContent) + { + yield return new ToolCallResultEvent + { + MessageId = chatResponse.MessageId, + ToolCallId = functionResultContent.CallId, + Content = SerializeResultContent(functionResultContent, jsonSerializerOptions) ?? "", + Role = AGUIRoles.Tool + }; + } + else if (content is DataContent dataContent) + { + if (MediaTypeHeaderValue.TryParse(dataContent.MediaType, out var mediaType) && mediaType.Equals(s_json)) + { + // State snapshot event + yield return new StateSnapshotEvent + { +#if !NET + Snapshot = (JsonElement?)JsonSerializer.Deserialize( + dataContent.Data.ToArray(), + jsonSerializerOptions.GetTypeInfo(typeof(JsonElement))) +#else + Snapshot = (JsonElement?)JsonSerializer.Deserialize( + dataContent.Data.Span, + jsonSerializerOptions.GetTypeInfo(typeof(JsonElement))) +#endif + }; + } + else if (mediaType is { } && mediaType.Equals(s_jsonPatchMediaType)) + { + // State snapshot patch event must be a valid JSON patch, + // but its not up to us to validate that here. + yield return new StateDeltaEvent + { +#if !NET + Delta = (JsonElement?)JsonSerializer.Deserialize( + dataContent.Data.ToArray(), + jsonSerializerOptions.GetTypeInfo(typeof(JsonElement))) +#else + Delta = (JsonElement?)JsonSerializer.Deserialize( + dataContent.Data.Span, + jsonSerializerOptions.GetTypeInfo(typeof(JsonElement))) +#endif + }; + } + else + { + // Text content event + yield return new TextMessageContentEvent + { + MessageId = chatResponse.MessageId!, +#if !NET + Delta = Encoding.UTF8.GetString(dataContent.Data.ToArray()) +#else + Delta = Encoding.UTF8.GetString(dataContent.Data.Span) +#endif + }; + } + } + } + } + } + + // End the last message if there was one + if (currentMessageId is not null) + { + yield return new TextMessageEndEvent + { + MessageId = currentMessageId + }; + } + + yield return new RunFinishedEvent + { + ThreadId = threadId, + RunId = runId, + }; + } + + private static string? SerializeResultContent(FunctionResultContent functionResultContent, JsonSerializerOptions options) + { + return functionResultContent.Result switch + { + null => null, + string str => str, + JsonElement jsonElement => jsonElement.GetRawText(), + _ => JsonSerializer.Serialize(functionResultContent.Result, options.GetTypeInfo(functionResultContent.Result.GetType())), + }; + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/RunAgentInput.cs b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/RunAgentInput.cs index ad0d41cd8d..f64177146f 100644 --- a/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/RunAgentInput.cs +++ b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/RunAgentInput.cs @@ -1,6 +1,5 @@ // Copyright (c) Microsoft. All rights reserved. -using System; using System.Collections.Generic; using System.Text.Json; using System.Text.Json.Serialization; @@ -26,10 +25,14 @@ internal sealed class RunAgentInput [JsonPropertyName("messages")] public IEnumerable Messages { get; set; } = []; + [JsonPropertyName("tools")] + [JsonIgnore(Condition = JsonIgnoreCondition.WhenWritingDefault)] + public IEnumerable? Tools { get; set; } + [JsonPropertyName("context")] - public Dictionary Context { get; set; } = new(StringComparer.Ordinal); + public AGUIContextItem[] Context { get; set; } = []; - [JsonPropertyName("forwardedProperties")] + [JsonPropertyName("forwardedProps")] [JsonIgnore(Condition = JsonIgnoreCondition.WhenWritingDefault)] public JsonElement ForwardedProperties { get; set; } } diff --git a/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/StateDeltaEvent.cs b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/StateDeltaEvent.cs new file mode 100644 index 0000000000..98d3b168b3 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/StateDeltaEvent.cs @@ -0,0 +1,21 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json; +using System.Text.Json.Serialization; + +#if ASPNETCORE +namespace Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.Shared; +#else +namespace Microsoft.Agents.AI.AGUI.Shared; +#endif + +internal sealed class StateDeltaEvent : BaseEvent +{ + public StateDeltaEvent() + { + this.Type = AGUIEventTypes.StateDelta; + } + + [JsonPropertyName("delta")] + public JsonElement? Delta { get; set; } +} diff --git a/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/StateSnapshotEvent.cs b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/StateSnapshotEvent.cs new file mode 100644 index 0000000000..dc77e4ba46 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/StateSnapshotEvent.cs @@ -0,0 +1,21 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json; +using System.Text.Json.Serialization; + +#if ASPNETCORE +namespace Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.Shared; +#else +namespace Microsoft.Agents.AI.AGUI.Shared; +#endif + +internal sealed class StateSnapshotEvent : BaseEvent +{ + public StateSnapshotEvent() + { + this.Type = AGUIEventTypes.StateSnapshot; + } + + [JsonPropertyName("snapshot")] + public JsonElement? Snapshot { get; set; } +} diff --git a/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/ToolCallArgsEvent.cs b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/ToolCallArgsEvent.cs new file mode 100644 index 0000000000..27b0593699 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/ToolCallArgsEvent.cs @@ -0,0 +1,23 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +#if ASPNETCORE +namespace Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.Shared; +#else +namespace Microsoft.Agents.AI.AGUI.Shared; +#endif + +internal sealed class ToolCallArgsEvent : BaseEvent +{ + public ToolCallArgsEvent() + { + this.Type = AGUIEventTypes.ToolCallArgs; + } + + [JsonPropertyName("toolCallId")] + public string ToolCallId { get; set; } = string.Empty; + + [JsonPropertyName("delta")] + public string Delta { get; set; } = string.Empty; +} diff --git a/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/ToolCallEndEvent.cs b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/ToolCallEndEvent.cs new file mode 100644 index 0000000000..e78e6b89d9 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/ToolCallEndEvent.cs @@ -0,0 +1,20 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +#if ASPNETCORE +namespace Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.Shared; +#else +namespace Microsoft.Agents.AI.AGUI.Shared; +#endif + +internal sealed class ToolCallEndEvent : BaseEvent +{ + public ToolCallEndEvent() + { + this.Type = AGUIEventTypes.ToolCallEnd; + } + + [JsonPropertyName("toolCallId")] + public string ToolCallId { get; set; } = string.Empty; +} diff --git a/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/ToolCallResultEvent.cs b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/ToolCallResultEvent.cs new file mode 100644 index 0000000000..e60265be68 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/ToolCallResultEvent.cs @@ -0,0 +1,29 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +#if ASPNETCORE +namespace Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.Shared; +#else +namespace Microsoft.Agents.AI.AGUI.Shared; +#endif + +internal sealed class ToolCallResultEvent : BaseEvent +{ + public ToolCallResultEvent() + { + this.Type = AGUIEventTypes.ToolCallResult; + } + + [JsonPropertyName("messageId")] + public string? MessageId { get; set; } + + [JsonPropertyName("toolCallId")] + public string ToolCallId { get; set; } = string.Empty; + + [JsonPropertyName("content")] + public string Content { get; set; } = string.Empty; + + [JsonPropertyName("role")] + public string? Role { get; set; } +} diff --git a/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/ToolCallStartEvent.cs b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/ToolCallStartEvent.cs new file mode 100644 index 0000000000..e2f7bed120 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.AGUI/Shared/ToolCallStartEvent.cs @@ -0,0 +1,26 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +#if ASPNETCORE +namespace Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.Shared; +#else +namespace Microsoft.Agents.AI.AGUI.Shared; +#endif + +internal sealed class ToolCallStartEvent : BaseEvent +{ + public ToolCallStartEvent() + { + this.Type = AGUIEventTypes.ToolCallStart; + } + + [JsonPropertyName("toolCallId")] + public string ToolCallId { get; set; } = string.Empty; + + [JsonPropertyName("toolCallName")] + public string ToolCallName { get; set; } = string.Empty; + + [JsonPropertyName("parentMessageId")] + public string? ParentMessageId { get; set; } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Abstractions/AIAgent.cs b/dotnet/src/Microsoft.Agents.AI.Abstractions/AIAgent.cs index 35aa866552..4cff385dcc 100644 --- a/dotnet/src/Microsoft.Agents.AI.Abstractions/AIAgent.cs +++ b/dotnet/src/Microsoft.Agents.AI.Abstractions/AIAgent.cs @@ -22,9 +22,6 @@ namespace Microsoft.Agents.AI; [DebuggerDisplay("{DisplayName,nq}")] public abstract class AIAgent { - /// Default ID of this agent instance. - private readonly string _id = Guid.NewGuid().ToString("N"); - /// /// Gets the unique identifier for this agent instance. /// @@ -37,7 +34,19 @@ public abstract class AIAgent /// agent instances in multi-agent scenarios. They should remain stable for the lifetime /// of the agent instance. /// - public virtual string Id => this._id; + public string Id { get => this.IdCore ?? field; } = Guid.NewGuid().ToString("N"); + + /// + /// Gets a custom identifier for the agent, which can be overridden by derived classes. + /// + /// + /// A string representing the agent's identifier, or if the default ID should be used. + /// + /// + /// Derived classes can override this property to provide a custom identifier. + /// When is returned, the property will use the default randomly-generated identifier. + /// + protected virtual string? IdCore => null; /// /// Gets the human-readable name of the agent. @@ -61,7 +70,7 @@ public abstract class AIAgent /// This property provides a guaranteed non-null string suitable for display in user interfaces, /// logs, or other contexts where a readable identifier is needed. /// - public virtual string DisplayName => this.Name ?? this.Id ?? this._id; // final fallback to _id in case Id override returns null + public virtual string DisplayName => this.Name ?? this.Id; /// /// Gets a description of the agent's purpose, capabilities, or behavior. @@ -328,28 +337,4 @@ public abstract IAsyncEnumerable RunStreamingAsync( AgentThread? thread = null, AgentRunOptions? options = null, CancellationToken cancellationToken = default); - - /// - /// Notifies the specified thread about new messages that have been added to the conversation. - /// - /// The conversation thread to notify about the new messages. - /// The collection of new messages to report to the thread. - /// The to monitor for cancellation requests. The default is . - /// A task that represents the asynchronous notification operation. - /// or is . - /// - /// - /// This method ensures that conversation threads are kept informed about message additions, which - /// is important for threads that manage their own state, memory components, or derived context. - /// While all agent implementations should notify their threads, the specific actions taken by - /// each thread type may vary. - /// - /// - protected static async Task NotifyThreadOfNewMessagesAsync(AgentThread thread, IEnumerable messages, CancellationToken cancellationToken) - { - _ = Throw.IfNull(thread); - _ = Throw.IfNull(messages); - - await thread.MessagesReceivedAsync(messages, cancellationToken).ConfigureAwait(false); - } } diff --git a/dotnet/src/Microsoft.Agents.AI.Abstractions/AIContextProvider.cs b/dotnet/src/Microsoft.Agents.AI.Abstractions/AIContextProvider.cs index a4b3f5d956..fd3ff10fc2 100644 --- a/dotnet/src/Microsoft.Agents.AI.Abstractions/AIContextProvider.cs +++ b/dotnet/src/Microsoft.Agents.AI.Abstractions/AIContextProvider.cs @@ -124,7 +124,7 @@ public virtual JsonElement Serialize(JsonSerializerOptions? jsonSerializerOption /// that will be used. Context providers can use this information to determine what additional context /// should be provided for the invocation. /// - public class InvokingContext + public sealed class InvokingContext { /// /// Initializes a new instance of the class with the specified request messages. @@ -153,7 +153,7 @@ public InvokingContext(IEnumerable requestMessages) /// request messages that were used and the response messages that were generated. It also indicates /// whether the invocation succeeded or failed. /// - public class InvokedContext + public sealed class InvokedContext { /// /// Initializes a new instance of the class with the specified request messages. diff --git a/dotnet/src/Microsoft.Agents.AI.Abstractions/AgentRunOptions.cs b/dotnet/src/Microsoft.Agents.AI.Abstractions/AgentRunOptions.cs index c6a64915cf..9cd6d51680 100644 --- a/dotnet/src/Microsoft.Agents.AI.Abstractions/AgentRunOptions.cs +++ b/dotnet/src/Microsoft.Agents.AI.Abstractions/AgentRunOptions.cs @@ -1,6 +1,7 @@ // Copyright (c) Microsoft. All rights reserved. using System; +using Microsoft.Extensions.AI; using Microsoft.Shared.Diagnostics; namespace Microsoft.Agents.AI; @@ -32,6 +33,7 @@ public AgentRunOptions(AgentRunOptions options) _ = Throw.IfNull(options); this.ContinuationToken = options.ContinuationToken; this.AllowBackgroundResponses = options.AllowBackgroundResponses; + this.AdditionalProperties = options.AdditionalProperties?.Clone(); } /// @@ -47,7 +49,7 @@ public AgentRunOptions(AgentRunOptions options) /// can be polled for completion by obtaining the token from the property /// and passing it via this property on subsequent calls to . /// - public object? ContinuationToken { get; set; } + public ResponseContinuationToken? ContinuationToken { get; set; } /// /// Gets or sets a value indicating whether the background responses are allowed. @@ -74,4 +76,18 @@ public AgentRunOptions(AgentRunOptions options) /// /// public bool? AllowBackgroundResponses { get; set; } + + /// + /// Gets or sets additional properties associated with these options. + /// + /// + /// An containing custom properties, + /// or if no additional properties are present. + /// + /// + /// Additional properties provide a way to include custom metadata or provider-specific + /// information that doesn't fit into the standard options schema. This is useful for + /// preserving implementation-specific details or extending the options with custom data. + /// + public AdditionalPropertiesDictionary? AdditionalProperties { get; set; } } diff --git a/dotnet/src/Microsoft.Agents.AI.Abstractions/AgentRunResponse.cs b/dotnet/src/Microsoft.Agents.AI.Abstractions/AgentRunResponse.cs index 2beb287918..001cfd9469 100644 --- a/dotnet/src/Microsoft.Agents.AI.Abstractions/AgentRunResponse.cs +++ b/dotnet/src/Microsoft.Agents.AI.Abstractions/AgentRunResponse.cs @@ -175,7 +175,7 @@ public IList Messages /// to poll for completion. /// /// - public object? ContinuationToken { get; set; } + public ResponseContinuationToken? ContinuationToken { get; set; } /// /// Gets or sets the timestamp indicating when this response was created. @@ -336,7 +336,7 @@ public bool TryDeserialize(JsonSerializerOptions serializerOptions, [NotNullW private static T? DeserializeFirstTopLevelObject(string json, JsonTypeInfo typeInfo) { -#if NET9_0_OR_GREATER +#if NET // We need to deserialize only the first top-level object as a workaround for a common LLM backend // issue. GPT 3.5 Turbo commonly returns multiple top-level objects after doing a function call. // See https://community.openai.com/t/2-json-objects-returned-when-using-function-calling-and-json-mode/574348 diff --git a/dotnet/src/Microsoft.Agents.AI.Abstractions/AgentRunResponseUpdate.cs b/dotnet/src/Microsoft.Agents.AI.Abstractions/AgentRunResponseUpdate.cs index 954893dbcb..ccf3deae54 100644 --- a/dotnet/src/Microsoft.Agents.AI.Abstractions/AgentRunResponseUpdate.cs +++ b/dotnet/src/Microsoft.Agents.AI.Abstractions/AgentRunResponseUpdate.cs @@ -38,9 +38,6 @@ public class AgentRunResponseUpdate /// The response update content items. private IList? _contents; - /// The name of the author of the update. - private string? _authorName; - /// Initializes a new instance of the class. [JsonConstructor] public AgentRunResponseUpdate() @@ -84,8 +81,8 @@ public AgentRunResponseUpdate(ChatResponseUpdate chatResponseUpdate) /// Gets or sets the name of the author of the response update. public string? AuthorName { - get => this._authorName; - set => this._authorName = string.IsNullOrWhiteSpace(value) ? null : value; + get => field; + set => field = string.IsNullOrWhiteSpace(value) ? null : value; } /// Gets or sets the role of the author of the response update. @@ -162,7 +159,7 @@ public IList Contents /// to resume streaming from the point of interruption. /// /// - public object? ContinuationToken { get; set; } + public ResponseContinuationToken? ContinuationToken { get; set; } /// public override string ToString() => this.Text; diff --git a/dotnet/src/Microsoft.Agents.AI.Abstractions/AgentThread.cs b/dotnet/src/Microsoft.Agents.AI.Abstractions/AgentThread.cs index fb5863a5c9..4794457f41 100644 --- a/dotnet/src/Microsoft.Agents.AI.Abstractions/AgentThread.cs +++ b/dotnet/src/Microsoft.Agents.AI.Abstractions/AgentThread.cs @@ -1,11 +1,7 @@ // Copyright (c) Microsoft. All rights reserved. using System; -using System.Collections.Generic; using System.Text.Json; -using System.Threading; -using System.Threading.Tasks; -using Microsoft.Extensions.AI; using Microsoft.Shared.Diagnostics; namespace Microsoft.Agents.AI; @@ -65,19 +61,6 @@ protected AgentThread() public virtual JsonElement Serialize(JsonSerializerOptions? jsonSerializerOptions = null) => default; - /// - /// This method is called when new messages have been contributed to the chat by any participant. - /// - /// - /// Inheritors can use this method to update their context based on the new message. - /// - /// The new messages. - /// The to monitor for cancellation requests. The default is . - /// A task that completes when the context has been updated. - /// The thread has been deleted. - protected internal virtual Task MessagesReceivedAsync(IEnumerable newMessages, CancellationToken cancellationToken = default) - => Task.CompletedTask; - /// Asks the for an object of the specified type . /// The type of object being requested. /// An optional key that can be used to help identify the target service. diff --git a/dotnet/src/Microsoft.Agents.AI.Abstractions/DelegatingAIAgent.cs b/dotnet/src/Microsoft.Agents.AI.Abstractions/DelegatingAIAgent.cs index 353c82c996..72f1980b1c 100644 --- a/dotnet/src/Microsoft.Agents.AI.Abstractions/DelegatingAIAgent.cs +++ b/dotnet/src/Microsoft.Agents.AI.Abstractions/DelegatingAIAgent.cs @@ -54,7 +54,7 @@ protected DelegatingAIAgent(AIAgent innerAgent) protected AIAgent InnerAgent { get; } /// - public override string Id => this.InnerAgent.Id; + protected override string? IdCore => this.InnerAgent.Id; /// public override string? Name => this.InnerAgent.Name; diff --git a/dotnet/src/Microsoft.Agents.AI.Abstractions/InMemoryAgentThread.cs b/dotnet/src/Microsoft.Agents.AI.Abstractions/InMemoryAgentThread.cs index af6080a715..13fcc134f0 100644 --- a/dotnet/src/Microsoft.Agents.AI.Abstractions/InMemoryAgentThread.cs +++ b/dotnet/src/Microsoft.Agents.AI.Abstractions/InMemoryAgentThread.cs @@ -4,8 +4,6 @@ using System.Collections.Generic; using System.Diagnostics; using System.Text.Json; -using System.Threading; -using System.Threading.Tasks; using Microsoft.Extensions.AI; namespace Microsoft.Agents.AI; @@ -116,10 +114,6 @@ public override JsonElement Serialize(JsonSerializerOptions? jsonSerializerOptio public override object? GetService(Type serviceType, object? serviceKey = null) => base.GetService(serviceType, serviceKey) ?? this.MessageStore?.GetService(serviceType, serviceKey); - /// - protected internal override Task MessagesReceivedAsync(IEnumerable newMessages, CancellationToken cancellationToken = default) - => this.MessageStore.AddMessagesAsync(newMessages, cancellationToken); - [DebuggerBrowsable(DebuggerBrowsableState.Never)] private string DebuggerDisplay => $"Count = {this.MessageStore.Count}"; diff --git a/dotnet/src/Microsoft.Agents.AI.Abstractions/InMemoryChatMessageStore.cs b/dotnet/src/Microsoft.Agents.AI.Abstractions/InMemoryChatMessageStore.cs index 17d1cdce93..79d303207c 100644 --- a/dotnet/src/Microsoft.Agents.AI.Abstractions/InMemoryChatMessageStore.cs +++ b/dotnet/src/Microsoft.Agents.AI.Abstractions/InMemoryChatMessageStore.cs @@ -97,8 +97,9 @@ public InMemoryChatMessageStore(IChatReducer? chatReducer, JsonElement serialize if (serializedStoreState.ValueKind is JsonValueKind.Object) { + var jso = jsonSerializerOptions ?? AgentAbstractionsJsonUtilities.DefaultOptions; var state = serializedStoreState.Deserialize( - AgentAbstractionsJsonUtilities.DefaultOptions.GetTypeInfo(typeof(StoreState))) as StoreState; + jso.GetTypeInfo(typeof(StoreState))) as StoreState; if (state?.Messages is { } messages) { this._messages = messages; @@ -164,7 +165,8 @@ public override JsonElement Serialize(JsonSerializerOptions? jsonSerializerOptio Messages = this._messages, }; - return JsonSerializer.SerializeToElement(state, AgentAbstractionsJsonUtilities.DefaultOptions.GetTypeInfo(typeof(StoreState))); + var jso = jsonSerializerOptions ?? AgentAbstractionsJsonUtilities.DefaultOptions; + return JsonSerializer.SerializeToElement(state, jso.GetTypeInfo(typeof(StoreState))); } /// diff --git a/dotnet/src/Microsoft.Agents.AI.Abstractions/Microsoft.Agents.AI.Abstractions.csproj b/dotnet/src/Microsoft.Agents.AI.Abstractions/Microsoft.Agents.AI.Abstractions.csproj index 4add7f427c..6b6f9d44f2 100644 --- a/dotnet/src/Microsoft.Agents.AI.Abstractions/Microsoft.Agents.AI.Abstractions.csproj +++ b/dotnet/src/Microsoft.Agents.AI.Abstractions/Microsoft.Agents.AI.Abstractions.csproj @@ -1,8 +1,6 @@ - $(ProjectsTargetFrameworks) - $(ProjectsDebugTargetFrameworks) Microsoft.Agents.AI $(NoWarn);MEAI001 preview diff --git a/dotnet/src/Microsoft.Agents.AI.Anthropic/AnthropicBetaServiceExtensions.cs b/dotnet/src/Microsoft.Agents.AI.Anthropic/AnthropicBetaServiceExtensions.cs new file mode 100644 index 0000000000..6b4f872a63 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Anthropic/AnthropicBetaServiceExtensions.cs @@ -0,0 +1,103 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Microsoft.Agents.AI; +using Microsoft.Extensions.AI; +using Microsoft.Extensions.Logging; +using Microsoft.Shared.Diagnostics; + +namespace Anthropic.Services; + +/// +/// Provides extension methods for the class. +/// +public static class AnthropicBetaServiceExtensions +{ + /// + /// Specifies the default maximum number of tokens allowed for processing operations. + /// + public static int DefaultMaxTokens { get; set; } = 4096; + + /// + /// Creates a new AI agent using the specified model and options. + /// + /// The Anthropic beta service. + /// The model to use for chat completions. + /// The instructions for the AI agent. + /// The name of the AI agent. + /// The description of the AI agent. + /// The tools available to the AI agent. + /// The default maximum tokens for chat completions. Defaults to if not provided. + /// Provides a way to customize the creation of the underlying used by the agent. + /// Optional logger factory for enabling logging within the agent. + /// An optional to use for resolving services required by the instances being invoked. + /// The created AI agent. + public static ChatClientAgent CreateAIAgent( + this IBetaService betaService, + string model, + string? instructions = null, + string? name = null, + string? description = null, + IList? tools = null, + int? defaultMaxTokens = null, + Func? clientFactory = null, + ILoggerFactory? loggerFactory = null, + IServiceProvider? services = null) + { + var options = new ChatClientAgentOptions + { + Name = name, + Description = description, + }; + + if (!string.IsNullOrWhiteSpace(instructions)) + { + options.ChatOptions ??= new(); + options.ChatOptions.Instructions = instructions; + } + + if (tools is { Count: > 0 }) + { + options.ChatOptions ??= new(); + options.ChatOptions.Tools = tools; + } + + var chatClient = betaService.AsIChatClient(model, defaultMaxTokens ?? DefaultMaxTokens); + + if (clientFactory is not null) + { + chatClient = clientFactory(chatClient); + } + + return new ChatClientAgent(chatClient, options, loggerFactory, services); + } + + /// + /// Creates an AI agent from an using the Anthropic Chat Completion API. + /// + /// The Anthropic to use for the agent. + /// Full set of options to configure the agent. + /// Provides a way to customize the creation of the underlying used by the agent. + /// Optional logger factory for enabling logging within the agent. + /// An optional to use for resolving services required by the instances being invoked. + /// An instance backed by the Anthropic Chat Completion service. + /// Thrown when or is . + public static ChatClientAgent CreateAIAgent( + this IBetaService betaService, + ChatClientAgentOptions options, + Func? clientFactory = null, + ILoggerFactory? loggerFactory = null, + IServiceProvider? services = null) + { + Throw.IfNull(betaService); + Throw.IfNull(options); + + var chatClient = betaService.AsIChatClient(); + + if (clientFactory is not null) + { + chatClient = clientFactory(chatClient); + } + + return new ChatClientAgent(chatClient, options, loggerFactory, services); + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Anthropic/AnthropicClientExtensions.cs b/dotnet/src/Microsoft.Agents.AI.Anthropic/AnthropicClientExtensions.cs new file mode 100644 index 0000000000..b4b8e2bc1e --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Anthropic/AnthropicClientExtensions.cs @@ -0,0 +1,103 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Microsoft.Agents.AI; +using Microsoft.Extensions.AI; +using Microsoft.Extensions.Logging; +using Microsoft.Shared.Diagnostics; + +namespace Anthropic; + +/// +/// Provides extension methods for the class. +/// +public static class AnthropicClientExtensions +{ + /// + /// Specifies the default maximum number of tokens allowed for processing operations. + /// + public static int DefaultMaxTokens { get; set; } = 4096; + + /// + /// Creates a new AI agent using the specified model and options. + /// + /// An Anthropic to use with the agent.. + /// The model to use for chat completions. + /// The instructions for the AI agent. + /// The name of the AI agent. + /// The description of the AI agent. + /// The tools available to the AI agent. + /// The default maximum tokens for chat completions. Defaults to if not provided. + /// Provides a way to customize the creation of the underlying used by the agent. + /// Optional logger factory for enabling logging within the agent. + /// An optional to use for resolving services required by the instances being invoked. + /// The created AI agent. + public static ChatClientAgent CreateAIAgent( + this IAnthropicClient client, + string model, + string? instructions = null, + string? name = null, + string? description = null, + IList? tools = null, + int? defaultMaxTokens = null, + Func? clientFactory = null, + ILoggerFactory? loggerFactory = null, + IServiceProvider? services = null) + { + var options = new ChatClientAgentOptions + { + Name = name, + Description = description, + }; + + if (!string.IsNullOrWhiteSpace(instructions)) + { + options.ChatOptions ??= new(); + options.ChatOptions.Instructions = instructions; + } + + if (tools is { Count: > 0 }) + { + options.ChatOptions ??= new(); + options.ChatOptions.Tools = tools; + } + + var chatClient = client.AsIChatClient(model, defaultMaxTokens ?? DefaultMaxTokens); + + if (clientFactory is not null) + { + chatClient = clientFactory(chatClient); + } + + return new ChatClientAgent(chatClient, options, loggerFactory, services); + } + + /// + /// Creates an AI agent from an using the Anthropic Chat Completion API. + /// + /// An Anthropic to use with the agent.. + /// Full set of options to configure the agent. + /// Provides a way to customize the creation of the underlying used by the agent. + /// Optional logger factory for enabling logging within the agent. + /// An optional to use for resolving services required by the instances being invoked. + /// An instance backed by the Anthropic Chat Completion service. + /// Thrown when or is . + public static ChatClientAgent CreateAIAgent( + this IAnthropicClient client, + ChatClientAgentOptions options, + Func? clientFactory = null, + ILoggerFactory? loggerFactory = null, + IServiceProvider? services = null) + { + Throw.IfNull(client); + Throw.IfNull(options); + + var chatClient = client.AsIChatClient(); + + if (clientFactory is not null) + { + chatClient = clientFactory(chatClient); + } + + return new ChatClientAgent(chatClient, options, loggerFactory, services); + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Anthropic/AnthropicClientJsonContext.cs b/dotnet/src/Microsoft.Agents.AI.Anthropic/AnthropicClientJsonContext.cs new file mode 100644 index 0000000000..080745f148 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Anthropic/AnthropicClientJsonContext.cs @@ -0,0 +1,13 @@ +// Copyright (c) Microsoft. All rights reserved. + +#pragma warning disable CA1812 + +using System.Text.Json; +using System.Text.Json.Serialization; + +namespace Microsoft.Agents.AI.Anthropic; + +[JsonSerializable(typeof(JsonElement))] +[JsonSerializable(typeof(string))] +[JsonSerializable(typeof(Dictionary))] +internal sealed partial class AnthropicClientJsonContext : JsonSerializerContext; diff --git a/dotnet/src/Microsoft.Agents.AI.Anthropic/Microsoft.Agents.AI.Anthropic.csproj b/dotnet/src/Microsoft.Agents.AI.Anthropic/Microsoft.Agents.AI.Anthropic.csproj new file mode 100644 index 0000000000..60b90a0212 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Anthropic/Microsoft.Agents.AI.Anthropic.csproj @@ -0,0 +1,26 @@ + + + + preview + enable + true + + + + + + + + + + + + + + + + Microsoft Agent Framework Anthropic Agents + Provides Microsoft Agent Framework support for Anthropic Agents. + + + diff --git a/dotnet/src/Microsoft.Agents.AI.AzureAI.Persistent/Microsoft.Agents.AI.AzureAI.Persistent.csproj b/dotnet/src/Microsoft.Agents.AI.AzureAI.Persistent/Microsoft.Agents.AI.AzureAI.Persistent.csproj index 9fcbc4c83f..31785a8fa9 100644 --- a/dotnet/src/Microsoft.Agents.AI.AzureAI.Persistent/Microsoft.Agents.AI.AzureAI.Persistent.csproj +++ b/dotnet/src/Microsoft.Agents.AI.AzureAI.Persistent/Microsoft.Agents.AI.AzureAI.Persistent.csproj @@ -1,8 +1,6 @@ - + - $(ProjectsTargetFrameworks) - $(ProjectsDebugTargetFrameworks) preview enable @@ -20,8 +18,8 @@ - Microsoft Agent Framework AzureAI - Provides Microsoft Agent Framework support for Azure AI. + Microsoft Agent Framework AzureAI Persistent Agents + Provides Microsoft Agent Framework support for Azure AI Persistent Agents. diff --git a/dotnet/src/Microsoft.Agents.AI.AzureAI.Persistent/PersistentAgentsClientExtensions.cs b/dotnet/src/Microsoft.Agents.AI.AzureAI.Persistent/PersistentAgentsClientExtensions.cs index 1d5f228fcc..5ca1436587 100644 --- a/dotnet/src/Microsoft.Agents.AI.AzureAI.Persistent/PersistentAgentsClientExtensions.cs +++ b/dotnet/src/Microsoft.Agents.AI.AzureAI.Persistent/PersistentAgentsClientExtensions.cs @@ -17,15 +17,21 @@ public static class PersistentAgentsClientExtensions /// The response containing the persistent agent to be converted. Cannot be . /// The default to use when interacting with the agent. /// Provides a way to customize the creation of the underlying used by the agent. + /// An optional to use for resolving services required by the instances being invoked. /// A instance that can be used to perform operations on the persistent agent. - public static ChatClientAgent GetAIAgent(this PersistentAgentsClient persistentAgentsClient, Response persistentAgentResponse, ChatOptions? chatOptions = null, Func? clientFactory = null) + public static ChatClientAgent GetAIAgent( + this PersistentAgentsClient persistentAgentsClient, + Response persistentAgentResponse, + ChatOptions? chatOptions = null, + Func? clientFactory = null, + IServiceProvider? services = null) { if (persistentAgentResponse is null) { throw new ArgumentNullException(nameof(persistentAgentResponse)); } - return GetAIAgent(persistentAgentsClient, persistentAgentResponse.Value, chatOptions, clientFactory); + return GetAIAgent(persistentAgentsClient, persistentAgentResponse.Value, chatOptions, clientFactory, services); } /// @@ -35,8 +41,14 @@ public static ChatClientAgent GetAIAgent(this PersistentAgentsClient persistentA /// The persistent agent metadata to be converted. Cannot be . /// The default to use when interacting with the agent. /// Provides a way to customize the creation of the underlying used by the agent. + /// An optional to use for resolving services required by the instances being invoked. /// A instance that can be used to perform operations on the persistent agent. - public static ChatClientAgent GetAIAgent(this PersistentAgentsClient persistentAgentsClient, PersistentAgent persistentAgentMetadata, ChatOptions? chatOptions = null, Func? clientFactory = null) + public static ChatClientAgent GetAIAgent( + this PersistentAgentsClient persistentAgentsClient, + PersistentAgent persistentAgentMetadata, + ChatOptions? chatOptions = null, + Func? clientFactory = null, + IServiceProvider? services = null) { if (persistentAgentMetadata is null) { @@ -55,14 +67,19 @@ public static ChatClientAgent GetAIAgent(this PersistentAgentsClient persistentA chatClient = clientFactory(chatClient); } + if (!string.IsNullOrWhiteSpace(persistentAgentMetadata.Instructions) && chatOptions?.Instructions is null) + { + chatOptions ??= new ChatOptions(); + chatOptions.Instructions = persistentAgentMetadata.Instructions; + } + return new ChatClientAgent(chatClient, options: new() { Id = persistentAgentMetadata.Id, Name = persistentAgentMetadata.Name, Description = persistentAgentMetadata.Description, - Instructions = persistentAgentMetadata.Instructions, ChatOptions = chatOptions - }); + }, services: services); } /// @@ -73,6 +90,7 @@ public static ChatClientAgent GetAIAgent(this PersistentAgentsClient persistentA /// The ID of the server side agent to create a for. /// Options that should apply to all runs of the agent. /// Provides a way to customize the creation of the underlying used by the agent. + /// An optional to use for resolving services required by the instances being invoked. /// The to monitor for cancellation requests. The default is . /// A instance that can be used to perform operations on the persistent agent. public static ChatClientAgent GetAIAgent( @@ -80,6 +98,7 @@ public static ChatClientAgent GetAIAgent( string agentId, ChatOptions? chatOptions = null, Func? clientFactory = null, + IServiceProvider? services = null, CancellationToken cancellationToken = default) { if (persistentAgentsClient is null) @@ -93,7 +112,7 @@ public static ChatClientAgent GetAIAgent( } var persistentAgentResponse = persistentAgentsClient.Administration.GetAgent(agentId, cancellationToken); - return persistentAgentsClient.GetAIAgent(persistentAgentResponse, chatOptions, clientFactory); + return persistentAgentsClient.GetAIAgent(persistentAgentResponse, chatOptions, clientFactory, services); } /// @@ -104,6 +123,7 @@ public static ChatClientAgent GetAIAgent( /// The ID of the server side agent to create a for. /// Options that should apply to all runs of the agent. /// Provides a way to customize the creation of the underlying used by the agent. + /// An optional to use for resolving services required by the instances being invoked. /// The to monitor for cancellation requests. The default is . /// A instance that can be used to perform operations on the persistent agent. public static async Task GetAIAgentAsync( @@ -111,6 +131,7 @@ public static async Task GetAIAgentAsync( string agentId, ChatOptions? chatOptions = null, Func? clientFactory = null, + IServiceProvider? services = null, CancellationToken cancellationToken = default) { if (persistentAgentsClient is null) @@ -124,7 +145,7 @@ public static async Task GetAIAgentAsync( } var persistentAgentResponse = await persistentAgentsClient.Administration.GetAgentAsync(agentId, cancellationToken).ConfigureAwait(false); - return persistentAgentsClient.GetAIAgent(persistentAgentResponse, chatOptions, clientFactory); + return persistentAgentsClient.GetAIAgent(persistentAgentResponse, chatOptions, clientFactory, services); } /// @@ -134,16 +155,22 @@ public static async Task GetAIAgentAsync( /// The response containing the persistent agent to be converted. Cannot be . /// Full set of options to configure the agent. /// Provides a way to customize the creation of the underlying used by the agent. + /// An optional to use for resolving services required by the instances being invoked. /// A instance that can be used to perform operations on the persistent agent. /// Thrown when or is . - public static ChatClientAgent GetAIAgent(this PersistentAgentsClient persistentAgentsClient, Response persistentAgentResponse, ChatClientAgentOptions options, Func? clientFactory = null) + public static ChatClientAgent GetAIAgent( + this PersistentAgentsClient persistentAgentsClient, + Response persistentAgentResponse, + ChatClientAgentOptions options, + Func? clientFactory = null, + IServiceProvider? services = null) { if (persistentAgentResponse is null) { throw new ArgumentNullException(nameof(persistentAgentResponse)); } - return GetAIAgent(persistentAgentsClient, persistentAgentResponse.Value, options, clientFactory); + return GetAIAgent(persistentAgentsClient, persistentAgentResponse.Value, options, clientFactory, services); } /// @@ -153,9 +180,15 @@ public static ChatClientAgent GetAIAgent(this PersistentAgentsClient persistentA /// The persistent agent metadata to be converted. Cannot be . /// Full set of options to configure the agent. /// Provides a way to customize the creation of the underlying used by the agent. + /// An optional to use for resolving services required by the instances being invoked. /// A instance that can be used to perform operations on the persistent agent. /// Thrown when or is . - public static ChatClientAgent GetAIAgent(this PersistentAgentsClient persistentAgentsClient, PersistentAgent persistentAgentMetadata, ChatClientAgentOptions options, Func? clientFactory = null) + public static ChatClientAgent GetAIAgent( + this PersistentAgentsClient persistentAgentsClient, + PersistentAgent persistentAgentMetadata, + ChatClientAgentOptions options, + Func? clientFactory = null, + IServiceProvider? services = null) { if (persistentAgentMetadata is null) { @@ -179,19 +212,24 @@ public static ChatClientAgent GetAIAgent(this PersistentAgentsClient persistentA chatClient = clientFactory(chatClient); } + if (!string.IsNullOrWhiteSpace(persistentAgentMetadata.Instructions) && options.ChatOptions?.Instructions is null) + { + options.ChatOptions ??= new ChatOptions(); + options.ChatOptions.Instructions = persistentAgentMetadata.Instructions; + } + var agentOptions = new ChatClientAgentOptions() { Id = persistentAgentMetadata.Id, Name = options.Name ?? persistentAgentMetadata.Name, Description = options.Description ?? persistentAgentMetadata.Description, - Instructions = options.Instructions ?? persistentAgentMetadata.Instructions, ChatOptions = options.ChatOptions, AIContextProviderFactory = options.AIContextProviderFactory, ChatMessageStoreFactory = options.ChatMessageStoreFactory, UseProvidedChatClientAsIs = options.UseProvidedChatClientAsIs }; - return new ChatClientAgent(chatClient, agentOptions); + return new ChatClientAgent(chatClient, agentOptions, services: services); } /// @@ -201,6 +239,7 @@ public static ChatClientAgent GetAIAgent(this PersistentAgentsClient persistentA /// The ID of the server side agent to create a for. /// Full set of options to configure the agent. /// Provides a way to customize the creation of the underlying used by the agent. + /// An optional to use for resolving services required by the instances being invoked. /// The to monitor for cancellation requests. The default is . /// A instance that can be used to perform operations on the persistent agent. /// Thrown when or is . @@ -210,6 +249,7 @@ public static ChatClientAgent GetAIAgent( string agentId, ChatClientAgentOptions options, Func? clientFactory = null, + IServiceProvider? services = null, CancellationToken cancellationToken = default) { if (persistentAgentsClient is null) @@ -228,7 +268,7 @@ public static ChatClientAgent GetAIAgent( } var persistentAgentResponse = persistentAgentsClient.Administration.GetAgent(agentId, cancellationToken); - return persistentAgentsClient.GetAIAgent(persistentAgentResponse, options, clientFactory); + return persistentAgentsClient.GetAIAgent(persistentAgentResponse, options, clientFactory, services); } /// @@ -238,6 +278,7 @@ public static ChatClientAgent GetAIAgent( /// The ID of the server side agent to create a for. /// Full set of options to configure the agent. /// Provides a way to customize the creation of the underlying used by the agent. + /// An optional to use for resolving services required by the instances being invoked. /// The to monitor for cancellation requests. The default is . /// A instance that can be used to perform operations on the persistent agent. /// Thrown when or is . @@ -247,6 +288,7 @@ public static async Task GetAIAgentAsync( string agentId, ChatClientAgentOptions options, Func? clientFactory = null, + IServiceProvider? services = null, CancellationToken cancellationToken = default) { if (persistentAgentsClient is null) @@ -265,7 +307,7 @@ public static async Task GetAIAgentAsync( } var persistentAgentResponse = await persistentAgentsClient.Administration.GetAgentAsync(agentId, cancellationToken).ConfigureAwait(false); - return persistentAgentsClient.GetAIAgent(persistentAgentResponse, options, clientFactory); + return persistentAgentsClient.GetAIAgent(persistentAgentResponse, options, clientFactory, services); } /// @@ -283,6 +325,7 @@ public static async Task GetAIAgentAsync( /// The response format for the agent. /// The metadata for the agent. /// Provides a way to customize the creation of the underlying used by the agent. + /// An optional to use for resolving services required by the instances being invoked. /// The to monitor for cancellation requests. The default is . /// A instance that can be used to perform operations on the newly created agent. public static async Task CreateAIAgentAsync( @@ -298,6 +341,7 @@ public static async Task CreateAIAgentAsync( BinaryData? responseFormat = null, IReadOnlyDictionary? metadata = null, Func? clientFactory = null, + IServiceProvider? services = null, CancellationToken cancellationToken = default) { if (persistentAgentsClient is null) @@ -319,7 +363,7 @@ public static async Task CreateAIAgentAsync( cancellationToken: cancellationToken).ConfigureAwait(false); // Get a local proxy for the agent to work with. - return await persistentAgentsClient.GetAIAgentAsync(createPersistentAgentResponse.Value.Id, clientFactory: clientFactory, cancellationToken: cancellationToken).ConfigureAwait(false); + return await persistentAgentsClient.GetAIAgentAsync(createPersistentAgentResponse.Value.Id, clientFactory: clientFactory, services: services, cancellationToken: cancellationToken).ConfigureAwait(false); } /// @@ -337,6 +381,7 @@ public static async Task CreateAIAgentAsync( /// The response format for the agent. /// The metadata for the agent. /// Provides a way to customize the creation of the underlying used by the agent. + /// An optional to use for resolving services required by the instances being invoked. /// The to monitor for cancellation requests. The default is . /// A instance that can be used to perform operations on the newly created agent. public static ChatClientAgent CreateAIAgent( @@ -352,6 +397,7 @@ public static ChatClientAgent CreateAIAgent( BinaryData? responseFormat = null, IReadOnlyDictionary? metadata = null, Func? clientFactory = null, + IServiceProvider? services = null, CancellationToken cancellationToken = default) { if (persistentAgentsClient is null) @@ -373,7 +419,7 @@ public static ChatClientAgent CreateAIAgent( cancellationToken: cancellationToken); // Get a local proxy for the agent to work with. - return persistentAgentsClient.GetAIAgent(createPersistentAgentResponse.Value.Id, clientFactory: clientFactory, cancellationToken: cancellationToken); + return persistentAgentsClient.GetAIAgent(createPersistentAgentResponse.Value.Id, clientFactory: clientFactory, services: services, cancellationToken: cancellationToken); } /// @@ -383,6 +429,7 @@ public static ChatClientAgent CreateAIAgent( /// The model to be used by the agent. /// Full set of options to configure the agent. /// Provides a way to customize the creation of the underlying used by the agent. + /// An optional to use for resolving services required by the instances being invoked. /// The to monitor for cancellation requests. The default is . /// A instance that can be used to perform operations on the newly created agent. /// Thrown when or or is . @@ -392,6 +439,7 @@ public static ChatClientAgent CreateAIAgent( string model, ChatClientAgentOptions options, Func? clientFactory = null, + IServiceProvider? services = null, CancellationToken cancellationToken = default) { if (persistentAgentsClient is null) @@ -415,7 +463,7 @@ public static ChatClientAgent CreateAIAgent( model: model, name: options.Name, description: options.Description, - instructions: options.Instructions, + instructions: options.ChatOptions?.Instructions, tools: toolDefinitionsAndResources.ToolDefinitions, toolResources: toolDefinitionsAndResources.ToolResources, temperature: null, @@ -431,7 +479,7 @@ public static ChatClientAgent CreateAIAgent( } // Get a local proxy for the agent to work with. - return persistentAgentsClient.GetAIAgent(createPersistentAgentResponse.Value.Id, options, clientFactory: clientFactory, cancellationToken: cancellationToken); + return persistentAgentsClient.GetAIAgent(createPersistentAgentResponse.Value.Id, options, clientFactory: clientFactory, services: services, cancellationToken: cancellationToken); } /// @@ -441,6 +489,7 @@ public static ChatClientAgent CreateAIAgent( /// The model to be used by the agent. /// Full set of options to configure the agent. /// Provides a way to customize the creation of the underlying used by the agent. + /// An optional to use for resolving services required by the instances being invoked. /// The to monitor for cancellation requests. The default is . /// A instance that can be used to perform operations on the newly created agent. /// Thrown when or or is . @@ -450,6 +499,7 @@ public static async Task CreateAIAgentAsync( string model, ChatClientAgentOptions options, Func? clientFactory = null, + IServiceProvider? services = null, CancellationToken cancellationToken = default) { if (persistentAgentsClient is null) @@ -473,7 +523,7 @@ public static async Task CreateAIAgentAsync( model: model, name: options.Name, description: options.Description, - instructions: options.Instructions, + instructions: options.ChatOptions?.Instructions, tools: toolDefinitionsAndResources.ToolDefinitions, toolResources: toolDefinitionsAndResources.ToolResources, temperature: null, @@ -489,7 +539,7 @@ public static async Task CreateAIAgentAsync( } // Get a local proxy for the agent to work with. - return await persistentAgentsClient.GetAIAgentAsync(createPersistentAgentResponse.Value.Id, options, clientFactory: clientFactory, cancellationToken: cancellationToken).ConfigureAwait(false); + return await persistentAgentsClient.GetAIAgentAsync(createPersistentAgentResponse.Value.Id, options, clientFactory: clientFactory, services: services, cancellationToken: cancellationToken).ConfigureAwait(false); } private static (List? ToolDefinitions, ToolResources? ToolResources, List? FunctionToolsAndOtherTools) ConvertAIToolsToToolDefinitions(IList? tools) @@ -506,7 +556,7 @@ private static (List? ToolDefinitions, ToolResources? ToolResour { case HostedCodeInterpreterTool codeTool: - toolDefinitions ??= new(); + toolDefinitions ??= []; toolDefinitions.Add(new CodeInterpreterToolDefinition()); if (codeTool.Inputs is { Count: > 0 }) @@ -527,7 +577,7 @@ private static (List? ToolDefinitions, ToolResources? ToolResour break; case HostedFileSearchTool fileSearchTool: - toolDefinitions ??= new(); + toolDefinitions ??= []; toolDefinitions.Add(new FileSearchToolDefinition { FileSearch = new() { MaxNumResults = fileSearchTool.MaximumResultCount } @@ -550,12 +600,12 @@ private static (List? ToolDefinitions, ToolResources? ToolResour break; case HostedWebSearchTool webSearch when webSearch.AdditionalProperties?.TryGetValue("connectionId", out object? connectionId) is true: - toolDefinitions ??= new(); + toolDefinitions ??= []; toolDefinitions.Add(new BingGroundingToolDefinition(new BingGroundingSearchToolParameters([new BingGroundingSearchConfiguration(connectionId!.ToString())]))); break; default: - functionToolsAndOtherTools ??= new(); + functionToolsAndOtherTools ??= []; functionToolsAndOtherTools.Add(tool); break; } diff --git a/dotnet/src/Microsoft.Agents.AI.AzureAI/AzureAIProjectChatClient.cs b/dotnet/src/Microsoft.Agents.AI.AzureAI/AzureAIProjectChatClient.cs new file mode 100644 index 0000000000..8acafc8fc3 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.AzureAI/AzureAIProjectChatClient.cs @@ -0,0 +1,148 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Runtime.CompilerServices; +using Azure.AI.Projects; +using Azure.AI.Projects.OpenAI; +using Microsoft.Extensions.AI; +using Microsoft.Shared.Diagnostics; +using OpenAI.Responses; + +#pragma warning disable OPENAI001 // Type is for evaluation purposes only and is subject to change or removal in future updates. Suppress this diagnostic to proceed. + +namespace Microsoft.Agents.AI.AzureAI; + +/// +/// Provides a chat client implementation that integrates with Azure AI Agents, enabling chat interactions using +/// Azure-specific agent capabilities. +/// +internal sealed class AzureAIProjectChatClient : DelegatingChatClient +{ + private readonly ChatClientMetadata? _metadata; + private readonly AIProjectClient _agentClient; + private readonly AgentVersion? _agentVersion; + private readonly AgentRecord? _agentRecord; + private readonly ChatOptions? _chatOptions; + private readonly AgentReference _agentReference; + /// + /// The usage of a no-op model is a necessary change to avoid OpenAIClients to throw exceptions when + /// used with Azure AI Agents as the model used is now defined at the agent creation time. + /// + private const string NoOpModel = "no-op"; + + /// + /// Initializes a new instance of the class. + /// + /// An instance of to interact with Azure AI Agents services. + /// An instance of representing the specific agent to use. + /// The default model to use for the agent, if applicable. + /// An instance of representing the options on how the agent was predefined. + /// + /// The provided should be decorated with a for proper functionality. + /// + internal AzureAIProjectChatClient(AIProjectClient aiProjectClient, AgentReference agentReference, string? defaultModelId, ChatOptions? chatOptions) + : base(Throw.IfNull(aiProjectClient) + .GetProjectOpenAIClient() + .GetOpenAIResponseClient(defaultModelId ?? NoOpModel) + .AsIChatClient()) + { + this._agentClient = aiProjectClient; + this._agentReference = Throw.IfNull(agentReference); + this._metadata = new ChatClientMetadata("azure.ai.agents", defaultModelId: defaultModelId); + this._chatOptions = chatOptions; + } + + /// + /// Initializes a new instance of the class. + /// + /// An instance of to interact with Azure AI Agents services. + /// An instance of representing the specific agent to use. + /// An instance of representing the options on how the agent was predefined. + /// + /// The provided should be decorated with a for proper functionality. + /// + internal AzureAIProjectChatClient(AIProjectClient aiProjectClient, AgentRecord agentRecord, ChatOptions? chatOptions) + : this(aiProjectClient, Throw.IfNull(agentRecord).Versions.Latest, chatOptions) + { + this._agentRecord = agentRecord; + } + + internal AzureAIProjectChatClient(AIProjectClient aiProjectClient, AgentVersion agentVersion, ChatOptions? chatOptions) + : this( + aiProjectClient, + new AgentReference(Throw.IfNull(agentVersion).Name, agentVersion.Version), + (agentVersion.Definition as PromptAgentDefinition)?.Model, + chatOptions) + { + this._agentVersion = agentVersion; + } + + /// + public override object? GetService(Type serviceType, object? serviceKey = null) + { + return (serviceKey is null && serviceType == typeof(ChatClientMetadata)) + ? this._metadata + : (serviceKey is null && serviceType == typeof(AIProjectClient)) + ? this._agentClient + : (serviceKey is null && serviceType == typeof(AgentVersion)) + ? this._agentVersion + : (serviceKey is null && serviceType == typeof(AgentRecord)) + ? this._agentRecord + : (serviceKey is null && serviceType == typeof(AgentReference)) + ? this._agentReference + : base.GetService(serviceType, serviceKey); + } + + /// + public override async Task GetResponseAsync(IEnumerable messages, ChatOptions? options = null, CancellationToken cancellationToken = default) + { + var agentOptions = this.GetAgentEnabledChatOptions(options); + + return await base.GetResponseAsync(messages, agentOptions, cancellationToken).ConfigureAwait(false); + } + + /// + public override async IAsyncEnumerable GetStreamingResponseAsync(IEnumerable messages, ChatOptions? options = null, [EnumeratorCancellation] CancellationToken cancellationToken = default) + { + var agentOptions = this.GetAgentEnabledChatOptions(options); + + await foreach (var chunk in base.GetStreamingResponseAsync(messages, agentOptions, cancellationToken).ConfigureAwait(false)) + { + yield return chunk; + } + } + + private ChatOptions GetAgentEnabledChatOptions(ChatOptions? options) + { + // Start with a clone of the base chat options defined for the agent, if any. + ChatOptions agentEnabledChatOptions = this._chatOptions?.Clone() ?? new(); + + // Ignore per-request all options that can't be overridden. + agentEnabledChatOptions.Instructions = null; + agentEnabledChatOptions.Tools = null; + agentEnabledChatOptions.Temperature = null; + agentEnabledChatOptions.TopP = null; + agentEnabledChatOptions.PresencePenalty = null; + agentEnabledChatOptions.ResponseFormat = null; + + // Use the conversation from the request, or the one defined at the client level. + agentEnabledChatOptions.ConversationId = options?.ConversationId ?? this._chatOptions?.ConversationId; + + // Preserve the original RawRepresentationFactory + var originalFactory = options?.RawRepresentationFactory; + + agentEnabledChatOptions.RawRepresentationFactory = (client) => + { + if (originalFactory?.Invoke(this) is not ResponseCreationOptions responseCreationOptions) + { + responseCreationOptions = new ResponseCreationOptions(); + } + + ResponseCreationOptionsExtensions.set_Agent(responseCreationOptions, this._agentReference); + ResponseCreationOptionsExtensions.set_Model(responseCreationOptions, null); + + return responseCreationOptions; + }; + + return agentEnabledChatOptions; + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.AzureAI/AzureAIProjectChatClientExtensions.cs b/dotnet/src/Microsoft.Agents.AI.AzureAI/AzureAIProjectChatClientExtensions.cs new file mode 100644 index 0000000000..dfbdad8e98 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.AzureAI/AzureAIProjectChatClientExtensions.cs @@ -0,0 +1,1038 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.ClientModel; +using System.ClientModel.Primitives; +using System.Runtime.CompilerServices; +using System.Text; +using System.Text.Json; +using System.Text.Json.Nodes; +using System.Text.Json.Serialization; +using System.Text.RegularExpressions; +using Azure.AI.Projects.OpenAI; +using Microsoft.Agents.AI; +using Microsoft.Agents.AI.AzureAI; +using Microsoft.Extensions.AI; +using Microsoft.Shared.Diagnostics; +using OpenAI; +using OpenAI.Responses; + +#pragma warning disable MEAI001 // Type is for evaluation purposes only and is subject to change or removal in future updates. Suppress this diagnostic to proceed. +#pragma warning disable OPENAI001 // Type is for evaluation purposes only and is subject to change or removal in future updates. Suppress this diagnostic to proceed. + +namespace Azure.AI.Projects; + +/// +/// Provides extension methods for . +/// +public static partial class AzureAIProjectChatClientExtensions +{ + /// + /// Retrieves an existing server side agent, wrapped as a using the provided . + /// + /// The to create the with. Cannot be . + /// The representing the name and version of the server side agent to create a for. Cannot be . + /// The tools to use when interacting with the agent. This is required when using prompt agent definitions with tools. + /// Provides a way to customize the creation of the underlying used by the agent. + /// An optional to use for resolving services required by the instances being invoked. + /// A instance that can be used to perform operations based on the latest version of the named Azure AI Agent. + /// Thrown when or is . + /// The agent with the specified name was not found. + /// + /// When retrieving an agent by using an , minimal information will be available about the agent in the instance level, and any logic that relies + /// on to retrieve information about the agent like will receive as the result. + /// + public static ChatClientAgent GetAIAgent( + this AIProjectClient aiProjectClient, + AgentReference agentReference, + IList? tools = null, + Func? clientFactory = null, + IServiceProvider? services = null) + { + Throw.IfNull(aiProjectClient); + Throw.IfNull(agentReference); + ThrowIfInvalidAgentName(agentReference.Name); + + return CreateChatClientAgent( + aiProjectClient, + agentReference, + new ChatClientAgentOptions() + { + Id = $"{agentReference.Name}:{agentReference.Version}", + Name = agentReference.Name, + ChatOptions = new() { Tools = tools }, + }, + clientFactory, + services); + } + + /// + /// Retrieves an existing server side agent, wrapped as a using the provided . + /// + /// The to create the with. Cannot be . + /// The name of the server side agent to create a for. Cannot be or whitespace. + /// The tools to use when interacting with the agent. This is required when using prompt agent definitions with tools. + /// Provides a way to customize the creation of the underlying used by the agent. + /// An optional to use for resolving services required by the instances being invoked. + /// The to monitor for cancellation requests. The default is . + /// A instance that can be used to perform operations based on the latest version of the named Azure AI Agent. + /// Thrown when or is . + /// Thrown when is empty or whitespace, or when the agent with the specified name was not found. + /// The agent with the specified name was not found. + public static ChatClientAgent GetAIAgent( + this AIProjectClient aiProjectClient, + string name, + IList? tools = null, + Func? clientFactory = null, + IServiceProvider? services = null, + CancellationToken cancellationToken = default) + { + Throw.IfNull(aiProjectClient); + ThrowIfInvalidAgentName(name); + + AgentRecord agentRecord = GetAgentRecordByName(aiProjectClient, name, cancellationToken); + + return GetAIAgent( + aiProjectClient, + agentRecord, + tools, + clientFactory, + services); + } + + /// + /// Asynchronously retrieves an existing server side agent, wrapped as a using the provided . + /// + /// The to create the with. Cannot be . + /// The name of the server side agent to create a for. Cannot be or whitespace. + /// The tools to use when interacting with the agent. This is required when using prompt agent definitions with tools. + /// Provides a way to customize the creation of the underlying used by the agent. + /// An optional to use for resolving services required by the instances being invoked. + /// The to monitor for cancellation requests. The default is . + /// A instance that can be used to perform operations based on the latest version of the named Azure AI Agent. + /// Thrown when or is . + /// Thrown when is empty or whitespace, or when the agent with the specified name was not found. + /// The agent with the specified name was not found. + public static async Task GetAIAgentAsync( + this AIProjectClient aiProjectClient, + string name, + IList? tools = null, + Func? clientFactory = null, + IServiceProvider? services = null, + CancellationToken cancellationToken = default) + { + Throw.IfNull(aiProjectClient); + ThrowIfInvalidAgentName(name); + + AgentRecord agentRecord = await GetAgentRecordByNameAsync(aiProjectClient, name, cancellationToken).ConfigureAwait(false); + + return GetAIAgent( + aiProjectClient, + agentRecord, + tools, + clientFactory, + services); + } + + /// + /// Gets a runnable agent instance from the provided agent record. + /// + /// The client used to interact with Azure AI Agents. Cannot be . + /// The agent record to be converted. The latest version will be used. Cannot be . + /// The tools to use when interacting with the agent. This is required when using prompt agent definitions with tools. + /// Provides a way to customize the creation of the underlying used by the agent. + /// An optional to use for resolving services required by the instances being invoked. + /// A instance that can be used to perform operations based on the latest version of the Azure AI Agent. + /// Thrown when or is . + public static ChatClientAgent GetAIAgent( + this AIProjectClient aiProjectClient, + AgentRecord agentRecord, + IList? tools = null, + Func? clientFactory = null, + IServiceProvider? services = null) + { + Throw.IfNull(aiProjectClient); + Throw.IfNull(agentRecord); + + var allowDeclarativeMode = tools is not { Count: > 0 }; + + return CreateChatClientAgent( + aiProjectClient, + agentRecord, + tools, + clientFactory, + !allowDeclarativeMode, + services); + } + + /// + /// Gets a runnable agent instance from a containing metadata about an Azure AI Agent. + /// + /// The client used to interact with Azure AI Agents. Cannot be . + /// The agent version to be converted. Cannot be . + /// In-process invocable tools to be provided. If no tools are provided manual handling will be necessary to invoke in-process tools. + /// Provides a way to customize the creation of the underlying used by the agent. + /// An optional to use for resolving services required by the instances being invoked. + /// A instance that can be used to perform operations based on the provided version of the Azure AI Agent. + /// Thrown when or is . + public static ChatClientAgent GetAIAgent( + this AIProjectClient aiProjectClient, + AgentVersion agentVersion, + IList? tools = null, + Func? clientFactory = null, + IServiceProvider? services = null) + { + Throw.IfNull(aiProjectClient); + Throw.IfNull(agentVersion); + + var allowDeclarativeMode = tools is not { Count: > 0 }; + + return CreateChatClientAgent( + aiProjectClient, + agentVersion, + tools, + clientFactory, + !allowDeclarativeMode, + services); + } + + /// + /// Creates a new Prompt AI Agent using the provided and options. + /// + /// The client used to manage and interact with AI agents. Cannot be . + /// The options for creating the agent. Cannot be . + /// A factory function to customize the creation of the chat client used by the agent. + /// An optional to use for resolving services required by the instances being invoked. + /// A to cancel the operation if needed. + /// A instance that can be used to perform operations on the newly created agent. + /// Thrown when or is . + public static ChatClientAgent GetAIAgent( + this AIProjectClient aiProjectClient, + ChatClientAgentOptions options, + Func? clientFactory = null, + IServiceProvider? services = null, + CancellationToken cancellationToken = default) + { + Throw.IfNull(aiProjectClient); + Throw.IfNull(options); + + if (string.IsNullOrWhiteSpace(options.Name)) + { + throw new ArgumentException("Agent name must be provided in the options.Name property", nameof(options)); + } + + ThrowIfInvalidAgentName(options.Name); + + AgentRecord agentRecord = GetAgentRecordByName(aiProjectClient, options.Name, cancellationToken); + var agentVersion = agentRecord.Versions.Latest; + + var agentOptions = CreateChatClientAgentOptions(agentVersion, options, requireInvocableTools: true); + + return CreateChatClientAgent( + aiProjectClient, + agentVersion, + agentOptions, + clientFactory, + services); + } + + /// + /// Creates a new Prompt AI Agent using the provided and options. + /// + /// The client used to manage and interact with AI agents. Cannot be . + /// The options for creating the agent. Cannot be . + /// A factory function to customize the creation of the chat client used by the agent. + /// An optional to use for resolving services required by the instances being invoked. + /// A to cancel the operation if needed. + /// A instance that can be used to perform operations on the newly created agent. + /// Thrown when or is . + public static async Task GetAIAgentAsync( + this AIProjectClient aiProjectClient, + ChatClientAgentOptions options, + Func? clientFactory = null, + IServiceProvider? services = null, + CancellationToken cancellationToken = default) + { + Throw.IfNull(aiProjectClient); + Throw.IfNull(options); + + if (string.IsNullOrWhiteSpace(options.Name)) + { + throw new ArgumentException("Agent name must be provided in the options.Name property", nameof(options)); + } + + ThrowIfInvalidAgentName(options.Name); + + AgentRecord agentRecord = await GetAgentRecordByNameAsync(aiProjectClient, options.Name, cancellationToken).ConfigureAwait(false); + var agentVersion = agentRecord.Versions.Latest; + + var agentOptions = CreateChatClientAgentOptions(agentVersion, options, requireInvocableTools: true); + + return CreateChatClientAgent( + aiProjectClient, + agentVersion, + agentOptions, + clientFactory, + services); + } + + /// + /// Creates a new Prompt AI agent using the specified configuration parameters. + /// + /// The client used to manage and interact with AI agents. Cannot be . + /// The name for the agent. + /// The name of the model to use for the agent. Cannot be or whitespace. + /// The instructions that guide the agent's behavior. Cannot be or whitespace. + /// The description for the agent. + /// The tools to use when interacting with the agent, this is required when using prompt agent definitions with tools. + /// A factory function to customize the creation of the chat client used by the agent. + /// An optional to use for resolving services required by the instances being invoked. + /// A token to monitor for cancellation requests. + /// A instance that can be used to perform operations on the newly created agent. + /// Thrown when , , or is . + /// Thrown when or is empty or whitespace. + /// When using prompt agent definitions with tools the parameter needs to be provided. + public static ChatClientAgent CreateAIAgent( + this AIProjectClient aiProjectClient, + string name, + string model, + string instructions, + string? description = null, + IList? tools = null, + Func? clientFactory = null, + IServiceProvider? services = null, + CancellationToken cancellationToken = default) + { + Throw.IfNull(aiProjectClient); + ThrowIfInvalidAgentName(name); + Throw.IfNullOrWhitespace(model); + Throw.IfNullOrWhitespace(instructions); + + return CreateAIAgent( + aiProjectClient, + name, + tools, + new AgentVersionCreationOptions(new PromptAgentDefinition(model) { Instructions = instructions }) { Description = description }, + clientFactory, + services, + cancellationToken); + } + + /// + /// Creates a new Prompt AI agent using the specified configuration parameters. + /// + /// The client used to manage and interact with AI agents. Cannot be . + /// The name for the agent. + /// The name of the model to use for the agent. Cannot be or whitespace. + /// The instructions that guide the agent's behavior. Cannot be or whitespace. + /// The description for the agent. + /// The tools to use when interacting with the agent, this is required when using prompt agent definitions with tools. + /// A factory function to customize the creation of the chat client used by the agent. + /// An optional to use for resolving services required by the instances being invoked. + /// A token to monitor for cancellation requests. + /// A instance that can be used to perform operations on the newly created agent. + /// Thrown when , , or is . + /// Thrown when or is empty or whitespace. + /// When using prompt agent definitions with tools the parameter needs to be provided. + public static Task CreateAIAgentAsync( + this AIProjectClient aiProjectClient, + string name, + string model, + string instructions, + string? description = null, + IList? tools = null, + Func? clientFactory = null, + IServiceProvider? services = null, + CancellationToken cancellationToken = default) + { + Throw.IfNull(aiProjectClient); + ThrowIfInvalidAgentName(name); + Throw.IfNullOrWhitespace(model); + Throw.IfNullOrWhitespace(instructions); + + return CreateAIAgentAsync( + aiProjectClient, + name, + tools, + new AgentVersionCreationOptions(new PromptAgentDefinition(model) { Instructions = instructions }) { Description = description }, + clientFactory, + services, + cancellationToken); + } + + /// + /// Creates a new Prompt AI Agent using the provided and options. + /// + /// The client used to manage and interact with AI agents. Cannot be . + /// The name of the model to use for the agent. Cannot be or whitespace. + /// The options for creating the agent. Cannot be . + /// A factory function to customize the creation of the chat client used by the agent. + /// An optional to use for resolving services required by the instances being invoked. + /// A to cancel the operation if needed. + /// A instance that can be used to perform operations on the newly created agent. + /// Thrown when or is . + /// Thrown when is empty or whitespace, or when the agent name is not provided in the options. + public static ChatClientAgent CreateAIAgent( + this AIProjectClient aiProjectClient, + string model, + ChatClientAgentOptions options, + Func? clientFactory = null, + IServiceProvider? services = null, + CancellationToken cancellationToken = default) + { + Throw.IfNull(aiProjectClient); + Throw.IfNull(options); + Throw.IfNullOrWhitespace(model); + const bool RequireInvocableTools = true; + + if (string.IsNullOrWhiteSpace(options.Name)) + { + throw new ArgumentException("Agent name must be provided in the options.Name property", nameof(options)); + } + + ThrowIfInvalidAgentName(options.Name); + + PromptAgentDefinition agentDefinition = new(model) + { + Instructions = options.ChatOptions?.Instructions, + Temperature = options.ChatOptions?.Temperature, + TopP = options.ChatOptions?.TopP, + TextOptions = new() { TextFormat = ToOpenAIResponseTextFormat(options.ChatOptions?.ResponseFormat, options.ChatOptions) } + }; + + // Attempt to capture breaking glass options from the raw representation factory that match the agent definition. + if (options.ChatOptions?.RawRepresentationFactory?.Invoke(new NoOpChatClient()) is ResponseCreationOptions respCreationOptions) + { + agentDefinition.ReasoningOptions = respCreationOptions.ReasoningOptions; + } + + ApplyToolsToAgentDefinition(agentDefinition, options.ChatOptions?.Tools); + + AgentVersionCreationOptions? creationOptions = new(agentDefinition); + if (!string.IsNullOrWhiteSpace(options.Description)) + { + creationOptions.Description = options.Description; + } + + AgentVersion agentVersion = CreateAgentVersionWithProtocol(aiProjectClient, options.Name, creationOptions, cancellationToken); + + var agentOptions = CreateChatClientAgentOptions(agentVersion, options, RequireInvocableTools); + + return CreateChatClientAgent( + aiProjectClient, + agentVersion, + agentOptions, + clientFactory, + services); + } + + /// + /// Creates a new Prompt AI Agent using the provided and options. + /// + /// The client used to manage and interact with AI agents. Cannot be . + /// The name of the model to use for the agent. Cannot be or whitespace. + /// The options for creating the agent. Cannot be . + /// A factory function to customize the creation of the chat client used by the agent. + /// An optional to use for resolving services required by the instances being invoked. + /// A to cancel the operation if needed. + /// A instance that can be used to perform operations on the newly created agent. + /// Thrown when or is . + /// Thrown when is empty or whitespace, or when the agent name is not provided in the options. + public static async Task CreateAIAgentAsync( + this AIProjectClient aiProjectClient, + string model, + ChatClientAgentOptions options, + Func? clientFactory = null, + IServiceProvider? services = null, + CancellationToken cancellationToken = default) + { + Throw.IfNull(aiProjectClient); + Throw.IfNull(options); + Throw.IfNullOrWhitespace(model); + const bool RequireInvocableTools = true; + + if (string.IsNullOrWhiteSpace(options.Name)) + { + throw new ArgumentException("Agent name must be provided in the options.Name property", nameof(options)); + } + + ThrowIfInvalidAgentName(options.Name); + + PromptAgentDefinition agentDefinition = new(model) + { + Instructions = options.ChatOptions?.Instructions, + Temperature = options.ChatOptions?.Temperature, + TopP = options.ChatOptions?.TopP, + TextOptions = new() { TextFormat = ToOpenAIResponseTextFormat(options.ChatOptions?.ResponseFormat, options.ChatOptions) } + }; + + // Attempt to capture breaking glass options from the raw representation factory that match the agent definition. + if (options.ChatOptions?.RawRepresentationFactory?.Invoke(new NoOpChatClient()) is ResponseCreationOptions respCreationOptions) + { + agentDefinition.ReasoningOptions = respCreationOptions.ReasoningOptions; + } + + ApplyToolsToAgentDefinition(agentDefinition, options.ChatOptions?.Tools); + + AgentVersionCreationOptions? creationOptions = new(agentDefinition); + if (!string.IsNullOrWhiteSpace(options.Description)) + { + creationOptions.Description = options.Description; + } + + AgentVersion agentVersion = await CreateAgentVersionWithProtocolAsync(aiProjectClient, options.Name, creationOptions, cancellationToken).ConfigureAwait(false); + + var agentOptions = CreateChatClientAgentOptions(agentVersion, options, RequireInvocableTools); + + return CreateChatClientAgent( + aiProjectClient, + agentVersion, + agentOptions, + clientFactory, + services); + } + + /// + /// Creates a new AI agent using the specified agent definition and optional configuration parameters. + /// + /// The client used to manage and interact with AI agents. Cannot be . + /// The name for the agent. + /// Settings that control the creation of the agent. + /// A factory function to customize the creation of the chat client used by the agent. + /// A token to monitor for cancellation requests. + /// A instance that can be used to perform operations on the newly created agent. + /// Thrown when or is . + /// + /// When using this extension method with a the tools are only declarative and not invocable. + /// Invocation of any in-process tools will need to be handled manually. + /// + public static ChatClientAgent CreateAIAgent( + this AIProjectClient aiProjectClient, + string name, + AgentVersionCreationOptions creationOptions, + Func? clientFactory = null, + CancellationToken cancellationToken = default) + { + Throw.IfNull(aiProjectClient); + ThrowIfInvalidAgentName(name); + Throw.IfNull(creationOptions); + + return CreateAIAgent( + aiProjectClient, + name, + tools: null, + creationOptions, + clientFactory, + services: null, + cancellationToken); + } + + /// + /// Asynchronously creates a new AI agent using the specified agent definition and optional configuration + /// parameters. + /// + /// The client used to manage and interact with AI agents. Cannot be . + /// The name for the agent. + /// Settings that control the creation of the agent. + /// A factory function to customize the creation of the chat client used by the agent. + /// A token to monitor for cancellation requests. + /// A instance that can be used to perform operations on the newly created agent. + /// Thrown when or is . + /// + /// When using this extension method with a the tools are only declarative and not invocable. + /// Invocation of any in-process tools will need to be handled manually. + /// + public static Task CreateAIAgentAsync( + this AIProjectClient aiProjectClient, + string name, + AgentVersionCreationOptions creationOptions, + Func? clientFactory = null, + CancellationToken cancellationToken = default) + { + Throw.IfNull(aiProjectClient); + ThrowIfInvalidAgentName(name); + Throw.IfNull(creationOptions); + + return CreateAIAgentAsync( + aiProjectClient, + name, + tools: null, + creationOptions, + clientFactory, + services: null, + cancellationToken); + } + + #region Private + + private static readonly ModelReaderWriterOptions s_modelWriterOptionsWire = new("W"); + + /// + /// Retrieves an agent record by name using the Protocol method with user-agent header. + /// + private static AgentRecord GetAgentRecordByName(AIProjectClient aiProjectClient, string agentName, CancellationToken cancellationToken) + { + ClientResult protocolResponse = aiProjectClient.Agents.GetAgent(agentName, cancellationToken.ToRequestOptions(false)); + var rawResponse = protocolResponse.GetRawResponse(); + AgentRecord? result = ModelReaderWriter.Read(rawResponse.Content, s_modelWriterOptionsWire, AzureAIProjectsOpenAIContext.Default); + return ClientResult.FromOptionalValue(result, rawResponse).Value! + ?? throw new InvalidOperationException($"Agent with name '{agentName}' not found."); + } + + /// + /// Asynchronously retrieves an agent record by name using the Protocol method with user-agent header. + /// + private static async Task GetAgentRecordByNameAsync(AIProjectClient aiProjectClient, string agentName, CancellationToken cancellationToken) + { + ClientResult protocolResponse = await aiProjectClient.Agents.GetAgentAsync(agentName, cancellationToken.ToRequestOptions(false)).ConfigureAwait(false); + var rawResponse = protocolResponse.GetRawResponse(); + AgentRecord? result = ModelReaderWriter.Read(rawResponse.Content, s_modelWriterOptionsWire, AzureAIProjectsOpenAIContext.Default); + return ClientResult.FromOptionalValue(result, rawResponse).Value! + ?? throw new InvalidOperationException($"Agent with name '{agentName}' not found."); + } + + /// + /// Creates an agent version using the Protocol method with user-agent header. + /// + private static AgentVersion CreateAgentVersionWithProtocol(AIProjectClient aiProjectClient, string agentName, AgentVersionCreationOptions creationOptions, CancellationToken cancellationToken) + { + using BinaryContent protocolRequest = BinaryContent.Create(ModelReaderWriter.Write(creationOptions, ModelReaderWriterOptions.Json, AzureAIProjectsContext.Default)); + ClientResult protocolResponse = aiProjectClient.Agents.CreateAgentVersion(agentName, protocolRequest, cancellationToken.ToRequestOptions(false)); + + var rawResponse = protocolResponse.GetRawResponse(); + AgentVersion? result = ModelReaderWriter.Read(rawResponse.Content, s_modelWriterOptionsWire, AzureAIProjectsOpenAIContext.Default); + return ClientResult.FromValue(result, rawResponse).Value!; + } + + /// + /// Asynchronously creates an agent version using the Protocol method with user-agent header. + /// + private static async Task CreateAgentVersionWithProtocolAsync(AIProjectClient aiProjectClient, string agentName, AgentVersionCreationOptions creationOptions, CancellationToken cancellationToken) + { + using BinaryContent protocolRequest = BinaryContent.Create(ModelReaderWriter.Write(creationOptions, ModelReaderWriterOptions.Json, AzureAIProjectsContext.Default)); + ClientResult protocolResponse = await aiProjectClient.Agents.CreateAgentVersionAsync(agentName, protocolRequest, cancellationToken.ToRequestOptions(false)).ConfigureAwait(false); + + var rawResponse = protocolResponse.GetRawResponse(); + AgentVersion? result = ModelReaderWriter.Read(rawResponse.Content, s_modelWriterOptionsWire, AzureAIProjectsOpenAIContext.Default); + return ClientResult.FromValue(result, rawResponse).Value!; + } + + private static ChatClientAgent CreateAIAgent( + this AIProjectClient aiProjectClient, + string name, + IList? tools, + AgentVersionCreationOptions creationOptions, + Func? clientFactory, + IServiceProvider? services, + CancellationToken cancellationToken) + { + var allowDeclarativeMode = tools is not { Count: > 0 }; + + if (!allowDeclarativeMode) + { + ApplyToolsToAgentDefinition(creationOptions.Definition, tools); + } + + AgentVersion agentVersion = CreateAgentVersionWithProtocol(aiProjectClient, name, creationOptions, cancellationToken); + + return CreateChatClientAgent( + aiProjectClient, + agentVersion, + tools, + clientFactory, + !allowDeclarativeMode, + services); + } + + private static async Task CreateAIAgentAsync( + this AIProjectClient aiProjectClient, + string name, + IList? tools, + AgentVersionCreationOptions creationOptions, + Func? clientFactory, + IServiceProvider? services, + CancellationToken cancellationToken) + { + var allowDeclarativeMode = tools is not { Count: > 0 }; + + if (!allowDeclarativeMode) + { + ApplyToolsToAgentDefinition(creationOptions.Definition, tools); + } + + AgentVersion agentVersion = await CreateAgentVersionWithProtocolAsync(aiProjectClient, name, creationOptions, cancellationToken).ConfigureAwait(false); + + return CreateChatClientAgent( + aiProjectClient, + agentVersion, + tools, + clientFactory, + !allowDeclarativeMode, + services); + } + + /// This method creates an with the specified ChatClientAgentOptions. + private static ChatClientAgent CreateChatClientAgent( + AIProjectClient aiProjectClient, + AgentVersion agentVersion, + ChatClientAgentOptions agentOptions, + Func? clientFactory, + IServiceProvider? services) + { + IChatClient chatClient = new AzureAIProjectChatClient(aiProjectClient, agentVersion, agentOptions.ChatOptions); + + if (clientFactory is not null) + { + chatClient = clientFactory(chatClient); + } + + return new ChatClientAgent(chatClient, agentOptions, services: services); + } + + /// This method creates an with the specified ChatClientAgentOptions. + private static ChatClientAgent CreateChatClientAgent( + AIProjectClient aiProjectClient, + AgentRecord agentRecord, + ChatClientAgentOptions agentOptions, + Func? clientFactory, + IServiceProvider? services) + { + IChatClient chatClient = new AzureAIProjectChatClient(aiProjectClient, agentRecord, agentOptions.ChatOptions); + + if (clientFactory is not null) + { + chatClient = clientFactory(chatClient); + } + + return new ChatClientAgent(chatClient, agentOptions, services: services); + } + + /// This method creates an with the specified ChatClientAgentOptions. + private static ChatClientAgent CreateChatClientAgent( + AIProjectClient aiProjectClient, + AgentReference agentReference, + ChatClientAgentOptions agentOptions, + Func? clientFactory, + IServiceProvider? services) + { + IChatClient chatClient = new AzureAIProjectChatClient(aiProjectClient, agentReference, defaultModelId: null, agentOptions.ChatOptions); + + if (clientFactory is not null) + { + chatClient = clientFactory(chatClient); + } + + return new ChatClientAgent(chatClient, agentOptions, services: services); + } + + /// This method creates an with a auto-generated ChatClientAgentOptions from the specified configuration parameters. + private static ChatClientAgent CreateChatClientAgent( + AIProjectClient AIProjectClient, + AgentVersion agentVersion, + IList? tools, + Func? clientFactory, + bool requireInvocableTools, + IServiceProvider? services) + => CreateChatClientAgent( + AIProjectClient, + agentVersion, + CreateChatClientAgentOptions(agentVersion, new ChatOptions() { Tools = tools }, requireInvocableTools), + clientFactory, + services); + + /// This method creates an with a auto-generated ChatClientAgentOptions from the specified configuration parameters. + private static ChatClientAgent CreateChatClientAgent( + AIProjectClient AIProjectClient, + AgentRecord agentRecord, + IList? tools, + Func? clientFactory, + bool requireInvocableTools, + IServiceProvider? services) + => CreateChatClientAgent( + AIProjectClient, + agentRecord, + CreateChatClientAgentOptions(agentRecord.Versions.Latest, new ChatOptions() { Tools = tools }, requireInvocableTools), + clientFactory, + services); + + /// + /// This method creates for the specified and the provided tools. + /// + /// The agent version. + /// The to use when interacting with the agent. + /// Indicates whether to enforce the presence of invocable tools when the AIAgent is created with an agent definition that uses them. + /// The created . + /// Thrown when the agent definition requires in-process tools but none were provided. + /// Thrown when the agent definition required tools were not provided. + /// + /// This method rebuilds the agent options from the agent definition returned by the version and combine with the in-proc tools when provided + /// this ensures that all required tools are provided and the definition of the agent options are consistent with the agent definition coming from the server. + /// + private static ChatClientAgentOptions CreateChatClientAgentOptions(AgentVersion agentVersion, ChatOptions? chatOptions, bool requireInvocableTools) + { + var agentDefinition = agentVersion.Definition; + + List? agentTools = null; + if (agentDefinition is PromptAgentDefinition { Tools: { Count: > 0 } definitionTools }) + { + // Check if no tools were provided while the agent definition requires in-proc tools. + if (requireInvocableTools && chatOptions?.Tools is not { Count: > 0 } && definitionTools.Any(t => t is FunctionTool)) + { + throw new ArgumentException("The agent definition in-process tools must be provided in the extension method tools parameter."); + } + + // Agregate all missing tools for a single error message. + List? missingTools = null; + + // Check function tools + foreach (ResponseTool responseTool in definitionTools) + { + if (requireInvocableTools && responseTool is FunctionTool functionTool) + { + // Check if a tool with the same type and name exists in the provided tools. + // When invocable tools are required, match only AIFunction. + var matchingTool = chatOptions?.Tools?.FirstOrDefault(t => t is AIFunction tf && functionTool.FunctionName == tf.Name); + + if (matchingTool is null) + { + (missingTools ??= []).Add($"Function tool: {functionTool.FunctionName}"); + } + else + { + (agentTools ??= []).Add(matchingTool!); + } + continue; + } + + (agentTools ??= []).Add(responseTool.AsAITool()); + } + + if (requireInvocableTools && missingTools is { Count: > 0 }) + { + throw new InvalidOperationException($"The following prompt agent definition required tools were not provided: {string.Join(", ", missingTools)}"); + } + } + + var agentOptions = new ChatClientAgentOptions() + { + Id = agentVersion.Id, + Name = agentVersion.Name, + Description = agentVersion.Description, + }; + + if (agentDefinition is PromptAgentDefinition promptAgentDefinition) + { + agentOptions.ChatOptions ??= chatOptions?.Clone() ?? new(); + agentOptions.ChatOptions.Instructions = promptAgentDefinition.Instructions; + agentOptions.ChatOptions.Temperature = promptAgentDefinition.Temperature; + agentOptions.ChatOptions.TopP = promptAgentDefinition.TopP; + } + + if (agentTools is { Count: > 0 }) + { + agentOptions.ChatOptions ??= chatOptions?.Clone() ?? new(); + agentOptions.ChatOptions.Tools = agentTools; + } + + return agentOptions; + } + + /// + /// Creates a new instance of configured for the specified agent version and + /// optional base options. + /// + /// The agent version to use when configuring the chat client agent options. + /// An optional instance whose relevant properties will be copied to the + /// returned options. If , only default values are used. + /// Specifies whether the returned options must include invocable tools. Set to to require + /// invocable tools; otherwise, . + /// A instance configured according to the specified parameters. + private static ChatClientAgentOptions CreateChatClientAgentOptions(AgentVersion agentVersion, ChatClientAgentOptions? options, bool requireInvocableTools) + { + var agentOptions = CreateChatClientAgentOptions(agentVersion, options?.ChatOptions, requireInvocableTools); + if (options is not null) + { + agentOptions.AIContextProviderFactory = options.AIContextProviderFactory; + agentOptions.ChatMessageStoreFactory = options.ChatMessageStoreFactory; + agentOptions.UseProvidedChatClientAsIs = options.UseProvidedChatClientAsIs; + } + + return agentOptions; + } + + /// + /// Adds the specified AI tools to a prompt agent definition, while also ensuring that all invocable tools are provided. + /// + /// The agent definition to which the tools will be applied. Must be a PromptAgentDefinition to support tools. + /// A list of AI tools to add to the agent definition. If null or empty, no tools are added. + /// Thrown if tools were provided but is not a . + /// When providing functions, they need to be invokable AIFunctions. + private static void ApplyToolsToAgentDefinition(AgentDefinition agentDefinition, IList? tools) + { + if (tools is { Count: > 0 }) + { + if (agentDefinition is not PromptAgentDefinition promptAgentDefinition) + { + throw new ArgumentException("Only prompt agent definitions support tools.", nameof(agentDefinition)); + } + + // When tools are provided, those should represent the complete set of tools for the agent definition. + // This is particularly important for existing agents so no duplication happens for what was already defined. + promptAgentDefinition.Tools.Clear(); + + foreach (var tool in tools) + { + // Ensure that any AIFunctions provided are In-Proc, not just the declarations. + if (tool is not AIFunction && ( + tool.GetService() is not null // Declarative FunctionTool converted as AsAITool() + || tool is AIFunctionDeclaration)) // AIFunctionDeclaration type + { + throw new InvalidOperationException("When providing functions, they need to be invokable AIFunctions. AIFunctions can be created correctly using AIFunctionFactory.Create"); + } + + promptAgentDefinition.Tools.Add( + // If this is a converted ResponseTool as AITool, we can directly retrieve the ResponseTool instance from GetService. + tool.GetService() + // Otherwise we should be able to convert existing MEAI Tool abstractions into OpenAI ResponseTools + ?? tool.AsOpenAIResponseTool() + ?? throw new InvalidOperationException("The provided AITool could not be converted to a ResponseTool, ensure that the AITool was created using responseTool.AsAITool() extension.")); + } + } + } + + private static ResponseTextFormat? ToOpenAIResponseTextFormat(ChatResponseFormat? format, ChatOptions? options = null) => + format switch + { + ChatResponseFormatText => ResponseTextFormat.CreateTextFormat(), + + ChatResponseFormatJson jsonFormat when StrictSchemaTransformCache.GetOrCreateTransformedSchema(jsonFormat) is { } jsonSchema => + ResponseTextFormat.CreateJsonSchemaFormat( + jsonFormat.SchemaName ?? "json_schema", + BinaryData.FromBytes(JsonSerializer.SerializeToUtf8Bytes(jsonSchema, AgentClientJsonContext.Default.JsonElement)), + jsonFormat.SchemaDescription, + HasStrict(options?.AdditionalProperties)), + + ChatResponseFormatJson => ResponseTextFormat.CreateJsonObjectFormat(), + + _ => null, + }; + + /// Key into AdditionalProperties used to store a strict option. + private const string StrictKey = "strictJsonSchema"; + + /// Gets whether the properties specify that strict schema handling is desired. + private static bool? HasStrict(IReadOnlyDictionary? additionalProperties) => + additionalProperties?.TryGetValue(StrictKey, out object? strictObj) is true && + strictObj is bool strictValue ? + strictValue : null; + + /// + /// Gets the JSON schema transformer cache conforming to OpenAI strict / structured output restrictions per + /// https://platform.openai.com/docs/guides/structured-outputs?api-mode=responses#supported-schemas. + /// + private static AIJsonSchemaTransformCache StrictSchemaTransformCache { get; } = new(new() + { + DisallowAdditionalProperties = true, + ConvertBooleanSchemas = true, + MoveDefaultKeywordToDescription = true, + RequireAllProperties = true, + TransformSchemaNode = (ctx, node) => + { + // Move content from common but unsupported properties to description. In particular, we focus on properties that + // the AIJsonUtilities schema generator might produce and/or that are explicitly mentioned in the OpenAI documentation. + + if (node is JsonObject schemaObj) + { + StringBuilder? additionalDescription = null; + + ReadOnlySpan unsupportedProperties = + [ + // Produced by AIJsonUtilities but not in allow list at https://platform.openai.com/docs/guides/structured-outputs#supported-properties: + "contentEncoding", "contentMediaType", "not", + + // Explicitly mentioned at https://platform.openai.com/docs/guides/structured-outputs?api-mode=responses#key-ordering as being unsupported with some models: + "minLength", "maxLength", "pattern", "format", + "minimum", "maximum", "multipleOf", + "patternProperties", + "minItems", "maxItems", + + // Explicitly mentioned at https://learn.microsoft.com/azure/ai-services/openai/how-to/structured-outputs?pivots=programming-language-csharp&tabs=python-secure%2Cdotnet-entra-id#unsupported-type-specific-keywords + // as being unsupported with Azure OpenAI: + "unevaluatedProperties", "propertyNames", "minProperties", "maxProperties", + "unevaluatedItems", "contains", "minContains", "maxContains", "uniqueItems", + ]; + + foreach (string propName in unsupportedProperties) + { + if (schemaObj[propName] is { } propNode) + { + _ = schemaObj.Remove(propName); + AppendLine(ref additionalDescription, propName, propNode); + } + } + + if (additionalDescription is not null) + { + schemaObj["description"] = schemaObj["description"] is { } descriptionNode && descriptionNode.GetValueKind() == JsonValueKind.String ? + $"{descriptionNode.GetValue()}{Environment.NewLine}{additionalDescription}" : + additionalDescription.ToString(); + } + + return node; + + static void AppendLine(ref StringBuilder? sb, string propName, JsonNode propNode) + { + sb ??= new(); + + if (sb.Length > 0) + { + _ = sb.AppendLine(); + } + + _ = sb.Append(propName).Append(": ").Append(propNode); + } + } + + return node; + }, + }); + + /// + /// This class is a no-op implementation of to be used to honor the argument passed + /// while triggering avoiding any unexpected exception on the caller implementation. + /// + private sealed class NoOpChatClient : IChatClient + { + public void Dispose() { } + + public Task GetResponseAsync(IEnumerable messages, ChatOptions? options = null, CancellationToken cancellationToken = default) + => Task.FromResult(new ChatResponse()); + + public object? GetService(Type serviceType, object? serviceKey = null) => null; + + public async IAsyncEnumerable GetStreamingResponseAsync(IEnumerable messages, ChatOptions? options = null, [EnumeratorCancellation] CancellationToken cancellationToken = default) + { + yield return new ChatResponseUpdate(); + } + } + #endregion + +#if NET + [GeneratedRegex("^[a-zA-Z0-9]([a-zA-Z0-9-]{0,61}[a-zA-Z0-9])?$")] + private static partial Regex AgentNameValidationRegex(); +#else + private static Regex AgentNameValidationRegex() => new("^[a-zA-Z0-9]([a-zA-Z0-9-]{0,61}[a-zA-Z0-9])?$"); +#endif + + private static string ThrowIfInvalidAgentName(string? name) + { + Throw.IfNullOrWhitespace(name); + if (!AgentNameValidationRegex().IsMatch(name)) + { + throw new ArgumentException("Agent name must be 1-63 characters long, start and end with an alphanumeric character, and can only contain alphanumeric characters or hyphens.", nameof(name)); + } + return name; + } +} + +[JsonSerializable(typeof(JsonElement))] +internal sealed partial class AgentClientJsonContext : JsonSerializerContext; diff --git a/dotnet/src/Microsoft.Agents.AI.AzureAI/Microsoft.Agents.AI.AzureAI.csproj b/dotnet/src/Microsoft.Agents.AI.AzureAI/Microsoft.Agents.AI.AzureAI.csproj new file mode 100644 index 0000000000..233718b3e4 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.AzureAI/Microsoft.Agents.AI.AzureAI.csproj @@ -0,0 +1,29 @@ + + + + preview + enable + true + + + + + + + + + + + + + + + + + + + Microsoft Agent Framework for Foundry Agents + Provides Microsoft Agent Framework support for Foundry Agents. + + + diff --git a/dotnet/src/Microsoft.Agents.AI.AzureAI/RequestOptionsExtensions.cs b/dotnet/src/Microsoft.Agents.AI.AzureAI/RequestOptionsExtensions.cs new file mode 100644 index 0000000000..722d316330 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.AzureAI/RequestOptionsExtensions.cs @@ -0,0 +1,67 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.ClientModel.Primitives; +using System.Reflection; + +namespace Microsoft.Agents.AI; + +internal static class RequestOptionsExtensions +{ + /// Creates a configured for use with Foundry Agents. + public static RequestOptions ToRequestOptions(this CancellationToken cancellationToken, bool streaming) + { + RequestOptions requestOptions = new() + { + CancellationToken = cancellationToken, + BufferResponse = !streaming + }; + + requestOptions.AddPolicy(MeaiUserAgentPolicy.Instance, PipelinePosition.PerCall); + + return requestOptions; + } + + /// Provides a pipeline policy that adds a "MEAI/x.y.z" user-agent header. + private sealed class MeaiUserAgentPolicy : PipelinePolicy + { + public static MeaiUserAgentPolicy Instance { get; } = new MeaiUserAgentPolicy(); + + private static readonly string s_userAgentValue = CreateUserAgentValue(); + + public override void Process(PipelineMessage message, IReadOnlyList pipeline, int currentIndex) + { + AddUserAgentHeader(message); + ProcessNext(message, pipeline, currentIndex); + } + + public override ValueTask ProcessAsync(PipelineMessage message, IReadOnlyList pipeline, int currentIndex) + { + AddUserAgentHeader(message); + return ProcessNextAsync(message, pipeline, currentIndex); + } + + private static void AddUserAgentHeader(PipelineMessage message) => + message.Request.Headers.Add("User-Agent", s_userAgentValue); + + private static string CreateUserAgentValue() + { + const string Name = "MEAI"; + + if (typeof(MeaiUserAgentPolicy).Assembly.GetCustomAttribute()?.InformationalVersion is string version) + { + int pos = version.IndexOf('+'); + if (pos >= 0) + { + version = version.Substring(0, pos); + } + + if (version.Length > 0) + { + return $"{Name}/{version}"; + } + } + + return Name; + } + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.CopilotStudio/ActivityProcessor.cs b/dotnet/src/Microsoft.Agents.AI.CopilotStudio/ActivityProcessor.cs index b4dfb59a28..178c8bc904 100644 --- a/dotnet/src/Microsoft.Agents.AI.CopilotStudio/ActivityProcessor.cs +++ b/dotnet/src/Microsoft.Agents.AI.CopilotStudio/ActivityProcessor.cs @@ -27,7 +27,7 @@ public static async IAsyncEnumerable ProcessActivityAsync(IAsyncEnu { yield return CreateChatMessageFromActivity(activity, [new TextContent(activity.Text)]); } - else + else if (logger.IsEnabled(LogLevel.Warning)) { logger.LogWarning("Unknown activity type '{ActivityType}' received.", activity.Type); } diff --git a/dotnet/src/Microsoft.Agents.AI.CopilotStudio/Microsoft.Agents.AI.CopilotStudio.csproj b/dotnet/src/Microsoft.Agents.AI.CopilotStudio/Microsoft.Agents.AI.CopilotStudio.csproj index d5aad73169..daa2757385 100644 --- a/dotnet/src/Microsoft.Agents.AI.CopilotStudio/Microsoft.Agents.AI.CopilotStudio.csproj +++ b/dotnet/src/Microsoft.Agents.AI.CopilotStudio/Microsoft.Agents.AI.CopilotStudio.csproj @@ -1,8 +1,6 @@ - $(ProjectsTargetFrameworks) - $(ProjectsDebugTargetFrameworks) preview diff --git a/dotnet/src/Microsoft.Agents.AI.CosmosNoSql/CosmosChatMessageStore.cs b/dotnet/src/Microsoft.Agents.AI.CosmosNoSql/CosmosChatMessageStore.cs new file mode 100644 index 0000000000..03334d90f9 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.CosmosNoSql/CosmosChatMessageStore.cs @@ -0,0 +1,688 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Collections.Generic; +using System.Diagnostics.CodeAnalysis; +using System.Linq; +using System.Text.Json; +using System.Threading; +using System.Threading.Tasks; +using Azure.Core; +using Microsoft.Azure.Cosmos; +using Microsoft.Extensions.AI; +using Microsoft.Shared.Diagnostics; + +namespace Microsoft.Agents.AI; + +/// +/// Provides a Cosmos DB implementation of the abstract class. +/// +[RequiresUnreferencedCode("The CosmosChatMessageStore uses JSON serialization which is incompatible with trimming.")] +[RequiresDynamicCode("The CosmosChatMessageStore uses JSON serialization which is incompatible with NativeAOT.")] +public sealed class CosmosChatMessageStore : ChatMessageStore, IDisposable +{ + private readonly CosmosClient _cosmosClient; + private readonly Container _container; + private readonly bool _ownsClient; + private bool _disposed; + + // Hierarchical partition key support + private readonly string? _tenantId; + private readonly string? _userId; + private readonly PartitionKey _partitionKey; + private readonly bool _useHierarchicalPartitioning; + + /// + /// Cached JSON serializer options for .NET 9.0 compatibility. + /// + private static readonly JsonSerializerOptions s_defaultJsonOptions = CreateDefaultJsonOptions(); + + private static JsonSerializerOptions CreateDefaultJsonOptions() + { + var options = new JsonSerializerOptions(); +#if NET9_0_OR_GREATER + // Configure TypeInfoResolver for .NET 9.0 to enable JSON serialization + options.TypeInfoResolver = new System.Text.Json.Serialization.Metadata.DefaultJsonTypeInfoResolver(); +#endif + return options; + } + + /// + /// Gets or sets the maximum number of messages to return in a single query batch. + /// Default is 100 for optimal performance. + /// + public int MaxItemCount { get; set; } = 100; + + /// + /// Gets or sets the maximum number of items per transactional batch operation. + /// Default is 100, maximum allowed by Cosmos DB is 100. + /// + public int MaxBatchSize { get; set; } = 100; + + /// + /// Gets or sets the maximum number of messages to retrieve from the store. + /// This helps prevent exceeding LLM context windows in long conversations. + /// Default is null (no limit). When set, only the most recent messages are returned. + /// + public int? MaxMessagesToRetrieve { get; set; } + + /// + /// Gets or sets the Time-To-Live (TTL) in seconds for messages. + /// Default is 86400 seconds (24 hours). Set to null to disable TTL. + /// + public int? MessageTtlSeconds { get; set; } = 86400; + + /// + /// Gets the conversation ID associated with this message store. + /// + public string ConversationId { get; init; } + + /// + /// Gets the database ID associated with this message store. + /// + public string DatabaseId { get; init; } + + /// + /// Gets the container ID associated with this message store. + /// + public string ContainerId { get; init; } + + /// + /// Internal primary constructor used by all public constructors. + /// + /// The instance to use for Cosmos DB operations. + /// The identifier of the Cosmos DB database. + /// The identifier of the Cosmos DB container. + /// The unique identifier for this conversation thread. + /// Whether this instance owns the CosmosClient and should dispose it. + /// Optional tenant identifier for hierarchical partitioning. + /// Optional user identifier for hierarchical partitioning. + internal CosmosChatMessageStore(CosmosClient cosmosClient, string databaseId, string containerId, string conversationId, bool ownsClient, string? tenantId = null, string? userId = null) + { + this._cosmosClient = Throw.IfNull(cosmosClient); + this._container = this._cosmosClient.GetContainer(Throw.IfNullOrWhitespace(databaseId), Throw.IfNullOrWhitespace(containerId)); + this.ConversationId = Throw.IfNullOrWhitespace(conversationId); + this.DatabaseId = databaseId; + this.ContainerId = containerId; + this._ownsClient = ownsClient; + + // Initialize partitioning mode + this._tenantId = tenantId; + this._userId = userId; + this._useHierarchicalPartitioning = tenantId != null && userId != null; + + this._partitionKey = this._useHierarchicalPartitioning + ? new PartitionKeyBuilder() + .Add(tenantId!) + .Add(userId!) + .Add(conversationId) + .Build() + : new PartitionKey(conversationId); + } + + /// + /// Initializes a new instance of the class using a connection string. + /// + /// The Cosmos DB connection string. + /// The identifier of the Cosmos DB database. + /// The identifier of the Cosmos DB container. + /// Thrown when any required parameter is null. + /// Thrown when any string parameter is null or whitespace. + public CosmosChatMessageStore(string connectionString, string databaseId, string containerId) + : this(connectionString, databaseId, containerId, Guid.NewGuid().ToString("N")) + { + } + + /// + /// Initializes a new instance of the class using a connection string. + /// + /// The Cosmos DB connection string. + /// The identifier of the Cosmos DB database. + /// The identifier of the Cosmos DB container. + /// The unique identifier for this conversation thread. + /// Thrown when any required parameter is null. + /// Thrown when any string parameter is null or whitespace. + public CosmosChatMessageStore(string connectionString, string databaseId, string containerId, string conversationId) + : this(new CosmosClient(Throw.IfNullOrWhitespace(connectionString)), databaseId, containerId, conversationId, ownsClient: true) + { + } + + /// + /// Initializes a new instance of the class using TokenCredential for authentication. + /// + /// The Cosmos DB account endpoint URI. + /// The TokenCredential to use for authentication (e.g., DefaultAzureCredential, ManagedIdentityCredential). + /// The identifier of the Cosmos DB database. + /// The identifier of the Cosmos DB container. + /// Thrown when any required parameter is null. + /// Thrown when any string parameter is null or whitespace. + public CosmosChatMessageStore(string accountEndpoint, TokenCredential tokenCredential, string databaseId, string containerId) + : this(accountEndpoint, tokenCredential, databaseId, containerId, Guid.NewGuid().ToString("N")) + { + } + + /// + /// Initializes a new instance of the class using a TokenCredential for authentication. + /// + /// The Cosmos DB account endpoint URI. + /// The TokenCredential to use for authentication (e.g., DefaultAzureCredential, ManagedIdentityCredential). + /// The identifier of the Cosmos DB database. + /// The identifier of the Cosmos DB container. + /// The unique identifier for this conversation thread. + /// Thrown when any required parameter is null. + /// Thrown when any string parameter is null or whitespace. + public CosmosChatMessageStore(string accountEndpoint, TokenCredential tokenCredential, string databaseId, string containerId, string conversationId) + : this(new CosmosClient(Throw.IfNullOrWhitespace(accountEndpoint), Throw.IfNull(tokenCredential)), databaseId, containerId, conversationId, ownsClient: true) + { + } + + /// + /// Initializes a new instance of the class using an existing . + /// + /// The instance to use for Cosmos DB operations. + /// The identifier of the Cosmos DB database. + /// The identifier of the Cosmos DB container. + /// Thrown when is null. + /// Thrown when any string parameter is null or whitespace. + public CosmosChatMessageStore(CosmosClient cosmosClient, string databaseId, string containerId) + : this(cosmosClient, databaseId, containerId, Guid.NewGuid().ToString("N")) + { + } + + /// + /// Initializes a new instance of the class using an existing . + /// + /// The instance to use for Cosmos DB operations. + /// The identifier of the Cosmos DB database. + /// The identifier of the Cosmos DB container. + /// The unique identifier for this conversation thread. + /// Thrown when is null. + /// Thrown when any string parameter is null or whitespace. + public CosmosChatMessageStore(CosmosClient cosmosClient, string databaseId, string containerId, string conversationId) + : this(cosmosClient, databaseId, containerId, conversationId, ownsClient: false) + { + } + + /// + /// Initializes a new instance of the class using a connection string with hierarchical partition keys. + /// + /// The Cosmos DB connection string. + /// The identifier of the Cosmos DB database. + /// The identifier of the Cosmos DB container. + /// The tenant identifier for hierarchical partitioning. + /// The user identifier for hierarchical partitioning. + /// The session identifier for hierarchical partitioning. + /// Thrown when any required parameter is null. + /// Thrown when any string parameter is null or whitespace. + public CosmosChatMessageStore(string connectionString, string databaseId, string containerId, string tenantId, string userId, string sessionId) + : this(new CosmosClient(Throw.IfNullOrWhitespace(connectionString)), databaseId, containerId, Throw.IfNullOrWhitespace(sessionId), ownsClient: true, Throw.IfNullOrWhitespace(tenantId), Throw.IfNullOrWhitespace(userId)) + { + } + + /// + /// Initializes a new instance of the class using a TokenCredential for authentication with hierarchical partition keys. + /// + /// The Cosmos DB account endpoint URI. + /// The TokenCredential to use for authentication (e.g., DefaultAzureCredential, ManagedIdentityCredential). + /// The identifier of the Cosmos DB database. + /// The identifier of the Cosmos DB container. + /// The tenant identifier for hierarchical partitioning. + /// The user identifier for hierarchical partitioning. + /// The session identifier for hierarchical partitioning. + /// Thrown when any required parameter is null. + /// Thrown when any string parameter is null or whitespace. + public CosmosChatMessageStore(string accountEndpoint, TokenCredential tokenCredential, string databaseId, string containerId, string tenantId, string userId, string sessionId) + : this(new CosmosClient(Throw.IfNullOrWhitespace(accountEndpoint), Throw.IfNull(tokenCredential)), databaseId, containerId, Throw.IfNullOrWhitespace(sessionId), ownsClient: true, Throw.IfNullOrWhitespace(tenantId), Throw.IfNullOrWhitespace(userId)) + { + } + + /// + /// Initializes a new instance of the class using an existing with hierarchical partition keys. + /// + /// The instance to use for Cosmos DB operations. + /// The identifier of the Cosmos DB database. + /// The identifier of the Cosmos DB container. + /// The tenant identifier for hierarchical partitioning. + /// The user identifier for hierarchical partitioning. + /// The session identifier for hierarchical partitioning. + /// Thrown when is null. + /// Thrown when any string parameter is null or whitespace. + public CosmosChatMessageStore(CosmosClient cosmosClient, string databaseId, string containerId, string tenantId, string userId, string sessionId) + : this(cosmosClient, databaseId, containerId, Throw.IfNullOrWhitespace(sessionId), ownsClient: false, Throw.IfNullOrWhitespace(tenantId), Throw.IfNullOrWhitespace(userId)) + { + } + + /// + /// Creates a new instance of the class from previously serialized state. + /// + /// The instance to use for Cosmos DB operations. + /// A representing the serialized state of the message store. + /// The identifier of the Cosmos DB database. + /// The identifier of the Cosmos DB container. + /// Optional settings for customizing the JSON deserialization process. + /// A new instance of initialized from the serialized state. + /// Thrown when is null. + /// Thrown when the serialized state cannot be deserialized. + public static CosmosChatMessageStore CreateFromSerializedState(CosmosClient cosmosClient, JsonElement serializedStoreState, string databaseId, string containerId, JsonSerializerOptions? jsonSerializerOptions = null) + { + Throw.IfNull(cosmosClient); + Throw.IfNullOrWhitespace(databaseId); + Throw.IfNullOrWhitespace(containerId); + + if (serializedStoreState.ValueKind is not JsonValueKind.Object) + { + throw new ArgumentException("Invalid serialized state", nameof(serializedStoreState)); + } + + var state = serializedStoreState.Deserialize(jsonSerializerOptions); + if (state?.ConversationIdentifier is not { } conversationId) + { + throw new ArgumentException("Invalid serialized state", nameof(serializedStoreState)); + } + + // Use the internal constructor with all parameters to ensure partition key logic is centralized + return state.UseHierarchicalPartitioning && state.TenantId != null && state.UserId != null + ? new CosmosChatMessageStore(cosmosClient, databaseId, containerId, conversationId, ownsClient: false, state.TenantId, state.UserId) + : new CosmosChatMessageStore(cosmosClient, databaseId, containerId, conversationId, ownsClient: false); + } + + /// + public override async Task> GetMessagesAsync(CancellationToken cancellationToken = default) + { +#pragma warning disable CA1513 // Use ObjectDisposedException.ThrowIf - not available on all target frameworks + if (this._disposed) + { + throw new ObjectDisposedException(this.GetType().FullName); + } +#pragma warning restore CA1513 + + // Fetch most recent messages in descending order when limit is set, then reverse to ascending + var orderDirection = this.MaxMessagesToRetrieve.HasValue ? "DESC" : "ASC"; + var query = new QueryDefinition($"SELECT * FROM c WHERE c.conversationId = @conversationId AND c.type = @type ORDER BY c.timestamp {orderDirection}") + .WithParameter("@conversationId", this.ConversationId) + .WithParameter("@type", "ChatMessage"); + + var iterator = this._container.GetItemQueryIterator(query, requestOptions: new QueryRequestOptions + { + PartitionKey = this._partitionKey, + MaxItemCount = this.MaxItemCount // Configurable query performance + }); + + var messages = new List(); + + while (iterator.HasMoreResults) + { + var response = await iterator.ReadNextAsync(cancellationToken).ConfigureAwait(false); + + foreach (var document in response) + { + if (this.MaxMessagesToRetrieve.HasValue && messages.Count >= this.MaxMessagesToRetrieve.Value) + { + break; + } + + if (!string.IsNullOrEmpty(document.Message)) + { + var message = JsonSerializer.Deserialize(document.Message, s_defaultJsonOptions); + if (message != null) + { + messages.Add(message); + } + } + } + + if (this.MaxMessagesToRetrieve.HasValue && messages.Count >= this.MaxMessagesToRetrieve.Value) + { + break; + } + } + + // If we fetched in descending order (most recent first), reverse to ascending order + if (this.MaxMessagesToRetrieve.HasValue) + { + messages.Reverse(); + } + + return messages; + } + + /// + public override async Task AddMessagesAsync(IEnumerable messages, CancellationToken cancellationToken = default) + { + if (messages is null) + { + throw new ArgumentNullException(nameof(messages)); + } + +#pragma warning disable CA1513 // Use ObjectDisposedException.ThrowIf - not available on all target frameworks + if (this._disposed) + { + throw new ObjectDisposedException(this.GetType().FullName); + } +#pragma warning restore CA1513 + + var messageList = messages as IReadOnlyCollection ?? messages.ToList(); + if (messageList.Count == 0) + { + return; + } + + // Use transactional batch for atomic operations + if (messageList.Count > 1) + { + await this.AddMessagesInBatchAsync(messageList, cancellationToken).ConfigureAwait(false); + } + else + { + await this.AddSingleMessageAsync(messageList.First(), cancellationToken).ConfigureAwait(false); + } + } + + /// + /// Adds multiple messages using transactional batch operations for atomicity. + /// + private async Task AddMessagesInBatchAsync(IReadOnlyCollection messages, CancellationToken cancellationToken) + { + var currentTimestamp = DateTimeOffset.UtcNow.ToUnixTimeSeconds(); + + // Process messages in optimal batch sizes + for (int i = 0; i < messages.Count; i += this.MaxBatchSize) + { + var batchMessages = messages.Skip(i).Take(this.MaxBatchSize).ToList(); + await this.ExecuteBatchOperationAsync(batchMessages, currentTimestamp, cancellationToken).ConfigureAwait(false); + } + } + + /// + /// Executes a single batch operation with enhanced error handling. + /// Cosmos SDK handles throttling (429) retries automatically. + /// + private async Task ExecuteBatchOperationAsync(List messages, long timestamp, CancellationToken cancellationToken) + { + // Create all documents upfront for validation and batch operation + var documents = new List(messages.Count); + foreach (var message in messages) + { + documents.Add(this.CreateMessageDocument(message, timestamp)); + } + + // Defensive check: Verify all messages share the same partition key values + // In hierarchical partitioning, this means same tenantId, userId, and sessionId + // In simple partitioning, this means same conversationId + if (documents.Count > 0) + { + if (this._useHierarchicalPartitioning) + { + // Verify all documents have matching hierarchical partition key components + var firstDoc = documents[0]; + if (!documents.All(d => d.TenantId == firstDoc.TenantId && d.UserId == firstDoc.UserId && d.SessionId == firstDoc.SessionId)) + { + throw new InvalidOperationException("All messages in a batch must share the same partition key values (tenantId, userId, sessionId)."); + } + } + else + { + // Verify all documents have matching conversationId + var firstConversationId = documents[0].ConversationId; + if (!documents.All(d => d.ConversationId == firstConversationId)) + { + throw new InvalidOperationException("All messages in a batch must share the same partition key value (conversationId)."); + } + } + } + + // All messages in this store share the same partition key by design + // Transactional batches require all items to share the same partition key + var batch = this._container.CreateTransactionalBatch(this._partitionKey); + + foreach (var document in documents) + { + batch.CreateItem(document); + } + + try + { + var response = await batch.ExecuteAsync(cancellationToken).ConfigureAwait(false); + if (!response.IsSuccessStatusCode) + { + throw new InvalidOperationException($"Batch operation failed with status: {response.StatusCode}. Details: {response.ErrorMessage}"); + } + } + catch (CosmosException ex) when (ex.StatusCode == System.Net.HttpStatusCode.RequestEntityTooLarge) + { + // If batch is too large, split into smaller batches + if (messages.Count == 1) + { + // Can't split further, use single operation + await this.AddSingleMessageAsync(messages[0], cancellationToken).ConfigureAwait(false); + return; + } + + // Split the batch in half and retry + var midpoint = messages.Count / 2; + var firstHalf = messages.Take(midpoint).ToList(); + var secondHalf = messages.Skip(midpoint).ToList(); + + await this.ExecuteBatchOperationAsync(firstHalf, timestamp, cancellationToken).ConfigureAwait(false); + await this.ExecuteBatchOperationAsync(secondHalf, timestamp, cancellationToken).ConfigureAwait(false); + } + } + + /// + /// Adds a single message to the store. + /// + private async Task AddSingleMessageAsync(ChatMessage message, CancellationToken cancellationToken) + { + var document = this.CreateMessageDocument(message, DateTimeOffset.UtcNow.ToUnixTimeSeconds()); + + try + { + await this._container.CreateItemAsync(document, this._partitionKey, cancellationToken: cancellationToken).ConfigureAwait(false); + } + catch (CosmosException ex) when (ex.StatusCode == System.Net.HttpStatusCode.RequestEntityTooLarge) + { + throw new InvalidOperationException( + "Message exceeds Cosmos DB's maximum item size limit of 2MB. " + + "Message ID: " + message.MessageId + ", Serialized size is too large. " + + "Consider reducing message content or splitting into smaller messages.", + ex); + } + } + + /// + /// Creates a message document with enhanced metadata. + /// + private CosmosMessageDocument CreateMessageDocument(ChatMessage message, long timestamp) + { + return new CosmosMessageDocument + { + Id = Guid.NewGuid().ToString(), + ConversationId = this.ConversationId, + Timestamp = timestamp, + MessageId = message.MessageId, + Role = message.Role.Value, + Message = JsonSerializer.Serialize(message, s_defaultJsonOptions), + Type = "ChatMessage", // Type discriminator + Ttl = this.MessageTtlSeconds, // Configurable TTL + // Include hierarchical metadata when using hierarchical partitioning + TenantId = this._useHierarchicalPartitioning ? this._tenantId : null, + UserId = this._useHierarchicalPartitioning ? this._userId : null, + SessionId = this._useHierarchicalPartitioning ? this.ConversationId : null + }; + } + + /// + public override JsonElement Serialize(JsonSerializerOptions? jsonSerializerOptions = null) + { +#pragma warning disable CA1513 // Use ObjectDisposedException.ThrowIf - not available on all target frameworks + if (this._disposed) + { + throw new ObjectDisposedException(this.GetType().FullName); + } +#pragma warning restore CA1513 + + var state = new StoreState + { + ConversationIdentifier = this.ConversationId, + TenantId = this._tenantId, + UserId = this._userId, + UseHierarchicalPartitioning = this._useHierarchicalPartitioning + }; + + var options = jsonSerializerOptions ?? s_defaultJsonOptions; + return JsonSerializer.SerializeToElement(state, options); + } + + /// + /// Gets the count of messages in this conversation. + /// This is an additional utility method beyond the base contract. + /// + /// The cancellation token. + /// The number of messages in the conversation. + public async Task GetMessageCountAsync(CancellationToken cancellationToken = default) + { +#pragma warning disable CA1513 // Use ObjectDisposedException.ThrowIf - not available on all target frameworks + if (this._disposed) + { + throw new ObjectDisposedException(this.GetType().FullName); + } +#pragma warning restore CA1513 + + // Efficient count query + var query = new QueryDefinition("SELECT VALUE COUNT(1) FROM c WHERE c.conversationId = @conversationId AND c.Type = @type") + .WithParameter("@conversationId", this.ConversationId) + .WithParameter("@type", "ChatMessage"); + + var iterator = this._container.GetItemQueryIterator(query, requestOptions: new QueryRequestOptions + { + PartitionKey = this._partitionKey + }); + + // COUNT queries always return a result + var response = await iterator.ReadNextAsync(cancellationToken).ConfigureAwait(false); + return response.FirstOrDefault(); + } + + /// + /// Deletes all messages in this conversation. + /// This is an additional utility method beyond the base contract. + /// + /// The cancellation token. + /// The number of messages deleted. + public async Task ClearMessagesAsync(CancellationToken cancellationToken = default) + { +#pragma warning disable CA1513 // Use ObjectDisposedException.ThrowIf - not available on all target frameworks + if (this._disposed) + { + throw new ObjectDisposedException(this.GetType().FullName); + } +#pragma warning restore CA1513 + + // Batch delete for efficiency + var query = new QueryDefinition("SELECT VALUE c.id FROM c WHERE c.conversationId = @conversationId AND c.Type = @type") + .WithParameter("@conversationId", this.ConversationId) + .WithParameter("@type", "ChatMessage"); + + var iterator = this._container.GetItemQueryIterator(query, requestOptions: new QueryRequestOptions + { + PartitionKey = this._partitionKey, + MaxItemCount = this.MaxItemCount + }); + + var deletedCount = 0; + + while (iterator.HasMoreResults) + { + var response = await iterator.ReadNextAsync(cancellationToken).ConfigureAwait(false); + var batch = this._container.CreateTransactionalBatch(this._partitionKey); + var batchItemCount = 0; + + foreach (var itemId in response) + { + if (!string.IsNullOrEmpty(itemId)) + { + batch.DeleteItem(itemId); + batchItemCount++; + deletedCount++; + } + } + + if (batchItemCount > 0) + { + await batch.ExecuteAsync(cancellationToken).ConfigureAwait(false); + } + } + + return deletedCount; + } + + /// + public void Dispose() + { + if (!this._disposed) + { + if (this._ownsClient) + { + this._cosmosClient?.Dispose(); + } + this._disposed = true; + } + } + + private sealed class StoreState + { + public string ConversationIdentifier { get; set; } = string.Empty; + public string? TenantId { get; set; } + public string? UserId { get; set; } + public bool UseHierarchicalPartitioning { get; set; } + } + + /// + /// Represents a document stored in Cosmos DB for chat messages. + /// + [SuppressMessage("Performance", "CA1812:Avoid uninstantiated internal classes", Justification = "Instantiated by Cosmos DB operations")] + private sealed class CosmosMessageDocument + { + [Newtonsoft.Json.JsonProperty("id")] + public string Id { get; set; } = string.Empty; + + [Newtonsoft.Json.JsonProperty("conversationId")] + public string ConversationId { get; set; } = string.Empty; + + [Newtonsoft.Json.JsonProperty("timestamp")] + public long Timestamp { get; set; } + + [Newtonsoft.Json.JsonProperty("messageId")] + public string? MessageId { get; set; } + + [Newtonsoft.Json.JsonProperty("role")] + public string? Role { get; set; } + + [Newtonsoft.Json.JsonProperty("message")] + public string Message { get; set; } = string.Empty; + + [Newtonsoft.Json.JsonProperty("type")] + public string Type { get; set; } = string.Empty; + + [Newtonsoft.Json.JsonProperty("ttl")] + public int? Ttl { get; set; } + + /// + /// Tenant ID for hierarchical partitioning scenarios (optional). + /// + [Newtonsoft.Json.JsonProperty("tenantId")] + public string? TenantId { get; set; } + + /// + /// User ID for hierarchical partitioning scenarios (optional). + /// + [Newtonsoft.Json.JsonProperty("userId")] + public string? UserId { get; set; } + + /// + /// Session ID for hierarchical partitioning scenarios (same as ConversationId for compatibility). + /// + [Newtonsoft.Json.JsonProperty("sessionId")] + public string? SessionId { get; set; } + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.CosmosNoSql/CosmosCheckpointStore.cs b/dotnet/src/Microsoft.Agents.AI.CosmosNoSql/CosmosCheckpointStore.cs new file mode 100644 index 0000000000..62987b1dfc --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.CosmosNoSql/CosmosCheckpointStore.cs @@ -0,0 +1,279 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Collections.Generic; +using System.Diagnostics.CodeAnalysis; +using System.Linq; +using System.Text.Json; +using System.Threading.Tasks; +using Azure.Core; +using Microsoft.Azure.Cosmos; +using Microsoft.Shared.Diagnostics; +using Newtonsoft.Json; +using Newtonsoft.Json.Linq; + +namespace Microsoft.Agents.AI.Workflows.Checkpointing; + +/// +/// Provides a Cosmos DB implementation of the abstract class. +/// +/// The type of objects to store as checkpoint values. +[RequiresUnreferencedCode("The CosmosCheckpointStore uses JSON serialization which is incompatible with trimming.")] +[RequiresDynamicCode("The CosmosCheckpointStore uses JSON serialization which is incompatible with NativeAOT.")] +public class CosmosCheckpointStore : JsonCheckpointStore, IDisposable +{ + private readonly CosmosClient _cosmosClient; + private readonly Container _container; + private readonly bool _ownsClient; + private bool _disposed; + + /// + /// Initializes a new instance of the class using a connection string. + /// + /// The Cosmos DB connection string. + /// The identifier of the Cosmos DB database. + /// The identifier of the Cosmos DB container. + /// Thrown when any required parameter is null. + /// Thrown when any string parameter is null or whitespace. + public CosmosCheckpointStore(string connectionString, string databaseId, string containerId) + { + var cosmosClientOptions = new CosmosClientOptions(); + + this._cosmosClient = new CosmosClient(Throw.IfNullOrWhitespace(connectionString), cosmosClientOptions); + this._container = this._cosmosClient.GetContainer(Throw.IfNullOrWhitespace(databaseId), Throw.IfNullOrWhitespace(containerId)); + this._ownsClient = true; + } + + /// + /// Initializes a new instance of the class using a TokenCredential for authentication. + /// + /// The Cosmos DB account endpoint URI. + /// The TokenCredential to use for authentication (e.g., DefaultAzureCredential, ManagedIdentityCredential). + /// The identifier of the Cosmos DB database. + /// The identifier of the Cosmos DB container. + /// Thrown when any required parameter is null. + /// Thrown when any string parameter is null or whitespace. + public CosmosCheckpointStore(string accountEndpoint, TokenCredential tokenCredential, string databaseId, string containerId) + { + var cosmosClientOptions = new CosmosClientOptions + { + SerializerOptions = new CosmosSerializationOptions + { + PropertyNamingPolicy = CosmosPropertyNamingPolicy.CamelCase + } + }; + + this._cosmosClient = new CosmosClient(Throw.IfNullOrWhitespace(accountEndpoint), Throw.IfNull(tokenCredential), cosmosClientOptions); + this._container = this._cosmosClient.GetContainer(Throw.IfNullOrWhitespace(databaseId), Throw.IfNullOrWhitespace(containerId)); + this._ownsClient = true; + } + + /// + /// Initializes a new instance of the class using an existing . + /// + /// The instance to use for Cosmos DB operations. + /// The identifier of the Cosmos DB database. + /// The identifier of the Cosmos DB container. + /// Thrown when is null. + /// Thrown when any string parameter is null or whitespace. + public CosmosCheckpointStore(CosmosClient cosmosClient, string databaseId, string containerId) + { + this._cosmosClient = Throw.IfNull(cosmosClient); + + this._container = this._cosmosClient.GetContainer(Throw.IfNullOrWhitespace(databaseId), Throw.IfNullOrWhitespace(containerId)); + this._ownsClient = false; + } + + /// + /// Gets the identifier of the Cosmos DB database. + /// + public string DatabaseId => this._container.Database.Id; + + /// + /// Gets the identifier of the Cosmos DB container. + /// + public string ContainerId => this._container.Id; + + /// + public override async ValueTask CreateCheckpointAsync(string runId, JsonElement value, CheckpointInfo? parent = null) + { + if (string.IsNullOrWhiteSpace(runId)) + { + throw new ArgumentException("Cannot be null or whitespace", nameof(runId)); + } + +#pragma warning disable CA1513 // Use ObjectDisposedException.ThrowIf - not available on all target frameworks + if (this._disposed) + { + throw new ObjectDisposedException(this.GetType().FullName); + } +#pragma warning restore CA1513 + + var checkpointId = Guid.NewGuid().ToString("N"); + var checkpointInfo = new CheckpointInfo(runId, checkpointId); + + var document = new CosmosCheckpointDocument + { + Id = $"{runId}_{checkpointId}", + RunId = runId, + CheckpointId = checkpointId, + Value = JToken.Parse(value.GetRawText()), + ParentCheckpointId = parent?.CheckpointId, + Timestamp = DateTimeOffset.UtcNow.ToUnixTimeSeconds() + }; + + await this._container.CreateItemAsync(document, new PartitionKey(runId)).ConfigureAwait(false); + return checkpointInfo; + } + + /// + public override async ValueTask RetrieveCheckpointAsync(string runId, CheckpointInfo key) + { + if (string.IsNullOrWhiteSpace(runId)) + { + throw new ArgumentException("Cannot be null or whitespace", nameof(runId)); + } + + if (key is null) + { + throw new ArgumentNullException(nameof(key)); + } + +#pragma warning disable CA1513 // Use ObjectDisposedException.ThrowIf - not available on all target frameworks + if (this._disposed) + { + throw new ObjectDisposedException(this.GetType().FullName); + } +#pragma warning restore CA1513 + + var id = $"{runId}_{key.CheckpointId}"; + + try + { + var response = await this._container.ReadItemAsync(id, new PartitionKey(runId)).ConfigureAwait(false); + using var document = JsonDocument.Parse(response.Resource.Value.ToString()); + return document.RootElement.Clone(); + } + catch (CosmosException ex) when (ex.StatusCode == System.Net.HttpStatusCode.NotFound) + { + throw new InvalidOperationException($"Checkpoint with ID '{key.CheckpointId}' for run '{runId}' not found."); + } + } + + /// + public override async ValueTask> RetrieveIndexAsync(string runId, CheckpointInfo? withParent = null) + { + if (string.IsNullOrWhiteSpace(runId)) + { + throw new ArgumentException("Cannot be null or whitespace", nameof(runId)); + } + +#pragma warning disable CA1513 // Use ObjectDisposedException.ThrowIf - not available on all target frameworks + if (this._disposed) + { + throw new ObjectDisposedException(this.GetType().FullName); + } +#pragma warning restore CA1513 + + QueryDefinition query = withParent == null + ? new QueryDefinition("SELECT c.runId, c.checkpointId FROM c WHERE c.runId = @runId ORDER BY c.timestamp ASC") + .WithParameter("@runId", runId) + : new QueryDefinition("SELECT c.runId, c.checkpointId FROM c WHERE c.runId = @runId AND c.parentCheckpointId = @parentCheckpointId ORDER BY c.timestamp ASC") + .WithParameter("@runId", runId) + .WithParameter("@parentCheckpointId", withParent.CheckpointId); + + var iterator = this._container.GetItemQueryIterator(query); + var checkpoints = new List(); + + while (iterator.HasMoreResults) + { + var response = await iterator.ReadNextAsync().ConfigureAwait(false); + checkpoints.AddRange(response.Select(r => new CheckpointInfo(r.RunId, r.CheckpointId))); + } + + return checkpoints; + } + + /// + public void Dispose() + { + this.Dispose(true); + GC.SuppressFinalize(this); + } + + /// + /// Releases the unmanaged resources used by the and optionally releases the managed resources. + /// + /// true to release both managed and unmanaged resources; false to release only unmanaged resources. + protected virtual void Dispose(bool disposing) + { + if (!this._disposed) + { + if (disposing && this._ownsClient) + { + this._cosmosClient?.Dispose(); + } + this._disposed = true; + } + } + + /// + /// Represents a checkpoint document stored in Cosmos DB. + /// + internal sealed class CosmosCheckpointDocument + { + [JsonProperty("id")] + public string Id { get; set; } = string.Empty; + + [JsonProperty("runId")] + public string RunId { get; set; } = string.Empty; + + [JsonProperty("checkpointId")] + public string CheckpointId { get; set; } = string.Empty; + + [JsonProperty("value")] + public JToken Value { get; set; } = JValue.CreateNull(); + + [JsonProperty("parentCheckpointId")] + public string? ParentCheckpointId { get; set; } + + [JsonProperty("timestamp")] + public long Timestamp { get; set; } + } + + /// + /// Represents the result of a checkpoint query. + /// + [SuppressMessage("Performance", "CA1812:Avoid uninstantiated internal classes", Justification = "Instantiated by Cosmos DB query deserialization")] + private sealed class CheckpointQueryResult + { + public string RunId { get; set; } = string.Empty; + public string CheckpointId { get; set; } = string.Empty; + } +} + +/// +/// Provides a non-generic Cosmos DB implementation of the abstract class. +/// +[RequiresUnreferencedCode("The CosmosCheckpointStore uses JSON serialization which is incompatible with trimming.")] +[RequiresDynamicCode("The CosmosCheckpointStore uses JSON serialization which is incompatible with NativeAOT.")] +public sealed class CosmosCheckpointStore : CosmosCheckpointStore +{ + /// + public CosmosCheckpointStore(string connectionString, string databaseId, string containerId) + : base(connectionString, databaseId, containerId) + { + } + + /// + public CosmosCheckpointStore(string accountEndpoint, TokenCredential tokenCredential, string databaseId, string containerId) + : base(accountEndpoint, tokenCredential, databaseId, containerId) + { + } + + /// + public CosmosCheckpointStore(CosmosClient cosmosClient, string databaseId, string containerId) + : base(cosmosClient, databaseId, containerId) + { + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.CosmosNoSql/CosmosDBChatExtensions.cs b/dotnet/src/Microsoft.Agents.AI.CosmosNoSql/CosmosDBChatExtensions.cs new file mode 100644 index 0000000000..4e3b66fd54 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.CosmosNoSql/CosmosDBChatExtensions.cs @@ -0,0 +1,95 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Diagnostics.CodeAnalysis; +using Azure.Identity; +using Microsoft.Azure.Cosmos; + +namespace Microsoft.Agents.AI; + +/// +/// Provides extension methods for integrating Cosmos DB chat message storage with the Agent Framework. +/// +public static class CosmosDBChatExtensions +{ + /// + /// Configures the agent to use Cosmos DB for message storage with connection string authentication. + /// + /// The chat client agent options to configure. + /// The Cosmos DB connection string. + /// The identifier of the Cosmos DB database. + /// The identifier of the Cosmos DB container. + /// The configured . + /// Thrown when is null. + /// Thrown when any string parameter is null or whitespace. + [RequiresUnreferencedCode("The CosmosChatMessageStore uses JSON serialization which is incompatible with trimming.")] + [RequiresDynamicCode("The CosmosChatMessageStore uses JSON serialization which is incompatible with NativeAOT.")] + public static ChatClientAgentOptions WithCosmosDBMessageStore( + this ChatClientAgentOptions options, + string connectionString, + string databaseId, + string containerId) + { + if (options is null) + { + throw new ArgumentNullException(nameof(options)); + } + + options.ChatMessageStoreFactory = context => new CosmosChatMessageStore(connectionString, databaseId, containerId); + return options; + } + + /// + /// Configures the agent to use Cosmos DB for message storage with managed identity authentication. + /// + /// The chat client agent options to configure. + /// The Cosmos DB account endpoint URI. + /// The identifier of the Cosmos DB database. + /// The identifier of the Cosmos DB container. + /// The configured . + /// Thrown when is null. + /// Thrown when any string parameter is null or whitespace. + [RequiresUnreferencedCode("The CosmosChatMessageStore uses JSON serialization which is incompatible with trimming.")] + [RequiresDynamicCode("The CosmosChatMessageStore uses JSON serialization which is incompatible with NativeAOT.")] + public static ChatClientAgentOptions WithCosmosDBMessageStoreUsingManagedIdentity( + this ChatClientAgentOptions options, + string accountEndpoint, + string databaseId, + string containerId) + { + if (options is null) + { + throw new ArgumentNullException(nameof(options)); + } + + options.ChatMessageStoreFactory = context => new CosmosChatMessageStore(accountEndpoint, new DefaultAzureCredential(), databaseId, containerId); + return options; + } + + /// + /// Configures the agent to use Cosmos DB for message storage with an existing . + /// + /// The chat client agent options to configure. + /// The instance to use for Cosmos DB operations. + /// The identifier of the Cosmos DB database. + /// The identifier of the Cosmos DB container. + /// The configured . + /// Thrown when any required parameter is null. + /// Thrown when any string parameter is null or whitespace. + [RequiresUnreferencedCode("The CosmosChatMessageStore uses JSON serialization which is incompatible with trimming.")] + [RequiresDynamicCode("The CosmosChatMessageStore uses JSON serialization which is incompatible with NativeAOT.")] + public static ChatClientAgentOptions WithCosmosDBMessageStore( + this ChatClientAgentOptions options, + CosmosClient cosmosClient, + string databaseId, + string containerId) + { + if (options is null) + { + throw new ArgumentNullException(nameof(options)); + } + + options.ChatMessageStoreFactory = context => new CosmosChatMessageStore(cosmosClient, databaseId, containerId); + return options; + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.CosmosNoSql/CosmosDBWorkflowExtensions.cs b/dotnet/src/Microsoft.Agents.AI.CosmosNoSql/CosmosDBWorkflowExtensions.cs new file mode 100644 index 0000000000..9d8bc52e68 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.CosmosNoSql/CosmosDBWorkflowExtensions.cs @@ -0,0 +1,218 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Diagnostics.CodeAnalysis; +using Azure.Identity; +using Microsoft.Agents.AI.Workflows.Checkpointing; +using Microsoft.Azure.Cosmos; + +namespace Microsoft.Agents.AI.Workflows; + +/// +/// Provides extension methods for integrating Cosmos DB checkpoint storage with the Agent Framework. +/// +public static class CosmosDBWorkflowExtensions +{ + /// + /// Creates a Cosmos DB checkpoint store using connection string authentication. + /// + /// The Cosmos DB connection string. + /// The identifier of the Cosmos DB database. + /// The identifier of the Cosmos DB container. + /// A new instance of . + /// Thrown when any string parameter is null or whitespace. + [RequiresUnreferencedCode("The CosmosCheckpointStore uses JSON serialization which is incompatible with trimming.")] + [RequiresDynamicCode("The CosmosCheckpointStore uses JSON serialization which is incompatible with NativeAOT.")] + public static CosmosCheckpointStore CreateCheckpointStore( + string connectionString, + string databaseId, + string containerId) + { + if (string.IsNullOrWhiteSpace(connectionString)) + { + throw new ArgumentException("Cannot be null or whitespace", nameof(connectionString)); + } + + if (string.IsNullOrWhiteSpace(databaseId)) + { + throw new ArgumentException("Cannot be null or whitespace", nameof(databaseId)); + } + + if (string.IsNullOrWhiteSpace(containerId)) + { + throw new ArgumentException("Cannot be null or whitespace", nameof(containerId)); + } + + return new CosmosCheckpointStore(connectionString, databaseId, containerId); + } + + /// + /// Creates a Cosmos DB checkpoint store using managed identity authentication. + /// + /// The Cosmos DB account endpoint URI. + /// The identifier of the Cosmos DB database. + /// The identifier of the Cosmos DB container. + /// A new instance of . + /// Thrown when any string parameter is null or whitespace. + [RequiresUnreferencedCode("The CosmosCheckpointStore uses JSON serialization which is incompatible with trimming.")] + [RequiresDynamicCode("The CosmosCheckpointStore uses JSON serialization which is incompatible with NativeAOT.")] + public static CosmosCheckpointStore CreateCheckpointStoreUsingManagedIdentity( + string accountEndpoint, + string databaseId, + string containerId) + { + if (string.IsNullOrWhiteSpace(accountEndpoint)) + { + throw new ArgumentException("Cannot be null or whitespace", nameof(accountEndpoint)); + } + + if (string.IsNullOrWhiteSpace(databaseId)) + { + throw new ArgumentException("Cannot be null or whitespace", nameof(databaseId)); + } + + if (string.IsNullOrWhiteSpace(containerId)) + { + throw new ArgumentException("Cannot be null or whitespace", nameof(containerId)); + } + + return new CosmosCheckpointStore(accountEndpoint, new DefaultAzureCredential(), databaseId, containerId); + } + + /// + /// Creates a Cosmos DB checkpoint store using an existing . + /// + /// The instance to use for Cosmos DB operations. + /// The identifier of the Cosmos DB database. + /// The identifier of the Cosmos DB container. + /// A new instance of . + /// Thrown when any required parameter is null. + /// Thrown when any string parameter is null or whitespace. + [RequiresUnreferencedCode("The CosmosCheckpointStore uses JSON serialization which is incompatible with trimming.")] + [RequiresDynamicCode("The CosmosCheckpointStore uses JSON serialization which is incompatible with NativeAOT.")] + public static CosmosCheckpointStore CreateCheckpointStore( + CosmosClient cosmosClient, + string databaseId, + string containerId) + { + if (cosmosClient is null) + { + throw new ArgumentNullException(nameof(cosmosClient)); + } + + if (string.IsNullOrWhiteSpace(databaseId)) + { + throw new ArgumentException("Cannot be null or whitespace", nameof(databaseId)); + } + + if (string.IsNullOrWhiteSpace(containerId)) + { + throw new ArgumentException("Cannot be null or whitespace", nameof(containerId)); + } + + return new CosmosCheckpointStore(cosmosClient, databaseId, containerId); + } + + /// + /// Creates a generic Cosmos DB checkpoint store using connection string authentication. + /// + /// The type of objects to store as checkpoint values. + /// The Cosmos DB connection string. + /// The identifier of the Cosmos DB database. + /// The identifier of the Cosmos DB container. + /// A new instance of . + /// Thrown when any string parameter is null or whitespace. + [RequiresUnreferencedCode("The CosmosCheckpointStore uses JSON serialization which is incompatible with trimming.")] + [RequiresDynamicCode("The CosmosCheckpointStore uses JSON serialization which is incompatible with NativeAOT.")] + public static CosmosCheckpointStore CreateCheckpointStore( + string connectionString, + string databaseId, + string containerId) + { + if (string.IsNullOrWhiteSpace(connectionString)) + { + throw new ArgumentException("Cannot be null or whitespace", nameof(connectionString)); + } + + if (string.IsNullOrWhiteSpace(databaseId)) + { + throw new ArgumentException("Cannot be null or whitespace", nameof(databaseId)); + } + + if (string.IsNullOrWhiteSpace(containerId)) + { + throw new ArgumentException("Cannot be null or whitespace", nameof(containerId)); + } + + return new CosmosCheckpointStore(connectionString, databaseId, containerId); + } + + /// + /// Creates a generic Cosmos DB checkpoint store using managed identity authentication. + /// + /// The type of objects to store as checkpoint values. + /// The Cosmos DB account endpoint URI. + /// The identifier of the Cosmos DB database. + /// The identifier of the Cosmos DB container. + /// A new instance of . + /// Thrown when any string parameter is null or whitespace. + [RequiresUnreferencedCode("The CosmosCheckpointStore uses JSON serialization which is incompatible with trimming.")] + [RequiresDynamicCode("The CosmosCheckpointStore uses JSON serialization which is incompatible with NativeAOT.")] + public static CosmosCheckpointStore CreateCheckpointStoreUsingManagedIdentity( + string accountEndpoint, + string databaseId, + string containerId) + { + if (string.IsNullOrWhiteSpace(accountEndpoint)) + { + throw new ArgumentException("Cannot be null or whitespace", nameof(accountEndpoint)); + } + + if (string.IsNullOrWhiteSpace(databaseId)) + { + throw new ArgumentException("Cannot be null or whitespace", nameof(databaseId)); + } + + if (string.IsNullOrWhiteSpace(containerId)) + { + throw new ArgumentException("Cannot be null or whitespace", nameof(containerId)); + } + + return new CosmosCheckpointStore(accountEndpoint, new DefaultAzureCredential(), databaseId, containerId); + } + + /// + /// Creates a generic Cosmos DB checkpoint store using an existing . + /// + /// The type of objects to store as checkpoint values. + /// The instance to use for Cosmos DB operations. + /// The identifier of the Cosmos DB database. + /// The identifier of the Cosmos DB container. + /// A new instance of . + /// Thrown when any required parameter is null. + /// Thrown when any string parameter is null or whitespace. + [RequiresUnreferencedCode("The CosmosCheckpointStore uses JSON serialization which is incompatible with trimming.")] + [RequiresDynamicCode("The CosmosCheckpointStore uses JSON serialization which is incompatible with NativeAOT.")] + public static CosmosCheckpointStore CreateCheckpointStore( + CosmosClient cosmosClient, + string databaseId, + string containerId) + { + if (cosmosClient is null) + { + throw new ArgumentNullException(nameof(cosmosClient)); + } + + if (string.IsNullOrWhiteSpace(databaseId)) + { + throw new ArgumentException("Cannot be null or whitespace", nameof(databaseId)); + } + + if (string.IsNullOrWhiteSpace(containerId)) + { + throw new ArgumentException("Cannot be null or whitespace", nameof(containerId)); + } + + return new CosmosCheckpointStore(cosmosClient, databaseId, containerId); + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.CosmosNoSql/Microsoft.Agents.AI.CosmosNoSql.csproj b/dotnet/src/Microsoft.Agents.AI.CosmosNoSql/Microsoft.Agents.AI.CosmosNoSql.csproj new file mode 100644 index 0000000000..7e13ec5998 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.CosmosNoSql/Microsoft.Agents.AI.CosmosNoSql.csproj @@ -0,0 +1,41 @@ + + + + $(TargetFrameworksCore) + Microsoft.Agents.AI + $(NoWarn);MEAI001 + preview + + + + true + true + true + true + true + true + + + + + + + Microsoft Agent Framework Cosmos DB NoSQL Integration + Provides Cosmos DB NoSQL implementations for Microsoft Agent Framework storage abstractions including ChatMessageStore and CheckpointStore. + + + + + + + + + + + + + + + + + diff --git a/dotnet/src/Microsoft.Agents.AI.Declarative/AgentBotElementYaml.cs b/dotnet/src/Microsoft.Agents.AI.Declarative/AgentBotElementYaml.cs new file mode 100644 index 0000000000..808bf76462 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Declarative/AgentBotElementYaml.cs @@ -0,0 +1,91 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Diagnostics.CodeAnalysis; +using System.IO; +using System.Linq; +using Microsoft.Bot.ObjectModel; +using Microsoft.Bot.ObjectModel.Abstractions; +using Microsoft.Bot.ObjectModel.Yaml; +using Microsoft.Extensions.Configuration; +using Microsoft.Shared.Diagnostics; + +namespace Microsoft.Agents.AI; + +/// +/// Helper methods for creating from YAML. +/// +internal static class AgentBotElementYaml +{ + /// + /// Convert the given YAML text to a model. + /// + /// YAML representation of the to use to create the prompt function. + /// Optional instance which provides environment variables to the template. + [RequiresDynamicCode("Calls YamlDotNet.Serialization.DeserializerBuilder.DeserializerBuilder()")] + public static GptComponentMetadata FromYaml(string text, IConfiguration? configuration = null) + { + Throw.IfNullOrEmpty(text); + + using var yamlReader = new StringReader(text); + BotElement rootElement = YamlSerializer.Deserialize(yamlReader) ?? throw new InvalidDataException("Text does not contain a valid agent definition."); + + if (rootElement is not GptComponentMetadata promptAgent) + { + throw new InvalidDataException($"Unsupported root element: {rootElement.GetType().Name}. Expected an {nameof(GptComponentMetadata)}."); + } + + var botDefinition = WrapPromptAgentWithBot(promptAgent, configuration); + + return botDefinition.Descendants().OfType().First(); + } + + #region private + private sealed class AgentFeatureConfiguration : IFeatureConfiguration + { + public long GetInt64Value(string settingName, long defaultValue) => defaultValue; + + public string GetStringValue(string settingName, string defaultValue) => defaultValue; + + public bool IsEnvironmentFeatureEnabled(string featureName, bool defaultValue) => true; + + public bool IsTenantFeatureEnabled(string featureName, bool defaultValue) => defaultValue; + } + + public static BotDefinition WrapPromptAgentWithBot(this GptComponentMetadata element, IConfiguration? configuration = null) + { + var botBuilder = + new BotDefinition.Builder + { + Components = + { + new GptComponent.Builder + { + SchemaName = "default-schema", + Metadata = element.ToBuilder(), + } + } + }; + + if (configuration is not null) + { + foreach (var kvp in configuration.AsEnumerable().Where(kvp => kvp.Value is not null)) + { + botBuilder.EnvironmentVariables.Add(new EnvironmentVariableDefinition.Builder() + { + SchemaName = kvp.Key, + Id = Guid.NewGuid(), + DisplayName = kvp.Key, + ValueComponent = new EnvironmentVariableValue.Builder() + { + Id = Guid.NewGuid(), + Value = kvp.Value!, + }, + }); + } + } + + return botBuilder.Build(); + } + #endregion +} diff --git a/dotnet/src/Microsoft.Agents.AI.Declarative/AggregatorPromptAgentFactory.cs b/dotnet/src/Microsoft.Agents.AI.Declarative/AggregatorPromptAgentFactory.cs new file mode 100644 index 0000000000..49027367f1 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Declarative/AggregatorPromptAgentFactory.cs @@ -0,0 +1,50 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Threading; +using System.Threading.Tasks; +using Microsoft.Bot.ObjectModel; +using Microsoft.Shared.Diagnostics; + +namespace Microsoft.Agents.AI; + +/// +/// Provides a which aggregates multiple agent factories. +/// +public sealed class AggregatorPromptAgentFactory : PromptAgentFactory +{ + private readonly PromptAgentFactory[] _agentFactories; + + /// Initializes the instance. + /// Ordered instances to aggregate. + /// + /// Where multiple instances are provided, the first factory that supports the will be used. + /// + public AggregatorPromptAgentFactory(params PromptAgentFactory[] agentFactories) + { + Throw.IfNullOrEmpty(agentFactories); + + foreach (PromptAgentFactory agentFactory in agentFactories) + { + Throw.IfNull(agentFactory, nameof(agentFactories)); + } + + this._agentFactories = agentFactories; + } + + /// + public override async Task TryCreateAsync(GptComponentMetadata promptAgent, CancellationToken cancellationToken = default) + { + Throw.IfNull(promptAgent); + + foreach (var agentFactory in this._agentFactories) + { + var agent = await agentFactory.TryCreateAsync(promptAgent, cancellationToken).ConfigureAwait(false); + if (agent is not null) + { + return agent; + } + } + + return null; + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Declarative/ChatClient/ChatClientPromptAgentFactory.cs b/dotnet/src/Microsoft.Agents.AI.Declarative/ChatClient/ChatClientPromptAgentFactory.cs new file mode 100644 index 0000000000..a7918de051 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Declarative/ChatClient/ChatClientPromptAgentFactory.cs @@ -0,0 +1,54 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Collections.Generic; +using System.Threading; +using System.Threading.Tasks; +using Microsoft.Bot.ObjectModel; +using Microsoft.Extensions.AI; +using Microsoft.Extensions.Configuration; +using Microsoft.Extensions.Logging; +using Microsoft.PowerFx; +using Microsoft.Shared.Diagnostics; + +namespace Microsoft.Agents.AI; + +/// +/// Provides an which creates instances of . +/// +public sealed class ChatClientPromptAgentFactory : PromptAgentFactory +{ + /// + /// Creates a new instance of the class. + /// + public ChatClientPromptAgentFactory(IChatClient chatClient, IList? functions = null, RecalcEngine? engine = null, IConfiguration? configuration = null, ILoggerFactory? loggerFactory = null) : base(engine, configuration) + { + Throw.IfNull(chatClient); + + this._chatClient = chatClient; + this._functions = functions; + this._loggerFactory = loggerFactory; + } + + /// + public override Task TryCreateAsync(GptComponentMetadata promptAgent, CancellationToken cancellationToken = default) + { + Throw.IfNull(promptAgent); + + var options = new ChatClientAgentOptions() + { + Name = promptAgent.Name, + Description = promptAgent.Description, + ChatOptions = promptAgent.GetChatOptions(this.Engine, this._functions), + }; + + var agent = new ChatClientAgent(this._chatClient, options, this._loggerFactory); + + return Task.FromResult(agent); + } + + #region private + private readonly IChatClient _chatClient; + private readonly IList? _functions; + private readonly ILoggerFactory? _loggerFactory; + #endregion +} diff --git a/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/BoolExpressionExtensions.cs b/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/BoolExpressionExtensions.cs new file mode 100644 index 0000000000..9926e0e6be --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/BoolExpressionExtensions.cs @@ -0,0 +1,56 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Microsoft.PowerFx; +using Microsoft.PowerFx.Types; + +namespace Microsoft.Bot.ObjectModel; + +/// +/// Extension methods for . +/// +internal static class BoolExpressionExtensions +{ + /// + /// Evaluates the given using the provided . + /// + /// Expression to evaluate. + /// Recalc engine to use for evaluation. + /// The evaluated boolean value, or null if the expression is null or cannot be evaluated. + internal static bool? Eval(this BoolExpression? expression, RecalcEngine? engine) + { + if (expression is null) + { + return null; + } + + if (expression.IsLiteral) + { + return expression.LiteralValue; + } + + if (engine is null) + { + return null; + } + + if (expression.IsExpression) + { + return engine.Eval(expression.ExpressionText!).AsBoolean(); + } + else if (expression.IsVariableReference) + { + var formulaValue = engine.Eval(expression.VariableReference!.VariableName); + if (formulaValue is BooleanValue booleanValue) + { + return booleanValue.Value; + } + + if (formulaValue is StringValue stringValue && bool.TryParse(stringValue.Value, out bool result)) + { + return result; + } + } + + return null; + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/CodeInterpreterToolExtensions.cs b/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/CodeInterpreterToolExtensions.cs new file mode 100644 index 0000000000..e6f13d5f54 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/CodeInterpreterToolExtensions.cs @@ -0,0 +1,23 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Microsoft.Extensions.AI; +using Microsoft.Shared.Diagnostics; + +namespace Microsoft.Bot.ObjectModel; + +/// +/// Extension methods for . +/// +internal static class CodeInterpreterToolExtensions +{ + /// + /// Creates a from a . + /// + /// Instance of + internal static HostedCodeInterpreterTool AsCodeInterpreterTool(this CodeInterpreterTool tool) + { + Throw.IfNull(tool); + + return new HostedCodeInterpreterTool(); + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/FileSearchToolExtensions.cs b/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/FileSearchToolExtensions.cs new file mode 100644 index 0000000000..5e1cb1bb5f --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/FileSearchToolExtensions.cs @@ -0,0 +1,28 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Linq; +using Microsoft.Extensions.AI; +using Microsoft.Shared.Diagnostics; + +namespace Microsoft.Bot.ObjectModel; + +/// +/// Extension methods for . +/// +internal static class FileSearchToolExtensions +{ + /// + /// Create a from a . + /// + /// Instance of + internal static HostedFileSearchTool CreateFileSearchTool(this FileSearchTool tool) + { + Throw.IfNull(tool); + + return new HostedFileSearchTool() + { + MaximumResultCount = (int?)tool.MaximumResultCount?.LiteralValue, + Inputs = tool.VectorStoreIds?.LiteralValue.Select(id => (AIContent)new HostedVectorStoreContent(id)).ToList(), + }; + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/FunctionToolExtensions.cs b/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/FunctionToolExtensions.cs new file mode 100644 index 0000000000..2c54d7e749 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/FunctionToolExtensions.cs @@ -0,0 +1,61 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Collections.Generic; +using System.Linq; +using System.Text.Json; +using Microsoft.Extensions.AI; +using Microsoft.Shared.Diagnostics; + +namespace Microsoft.Bot.ObjectModel; + +/// +/// Extension methods for . +/// +internal static class FunctionToolExtensions +{ + /// + /// Creates a from a . + /// + /// + /// If a matching function already exists in the provided list, it will be returned. + /// Otherwise, a new function declaration will be created. + /// + /// Instance of + /// Instance of + internal static AITool CreateOrGetAITool(this InvokeClientTaskAction tool, IList? functions) + { + Throw.IfNull(tool); + Throw.IfNull(tool.Name); + + // use the tool from the provided list if it exists + if (functions is not null) + { + var function = functions.FirstOrDefault(f => tool.Matches(f)); + + if (function is not null) + { + return function; + } + } + + return AIFunctionFactory.CreateDeclaration( + name: tool.Name, + description: tool.Description, + jsonSchema: tool.ClientActionInputSchema?.GetSchema() ?? s_defaultSchema); + } + + /// + /// Checks if a matches an . + /// + /// Instance of + /// Instance of + internal static bool Matches(this InvokeClientTaskAction tool, AIFunction aiFunc) + { + Throw.IfNull(tool); + Throw.IfNull(aiFunc); + + return tool.Name == aiFunc.Name; + } + + private static readonly JsonElement s_defaultSchema = JsonDocument.Parse("{\"type\":\"object\",\"properties\":{},\"additionalProperties\":false}").RootElement; +} diff --git a/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/IntExpressionExtensions.cs b/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/IntExpressionExtensions.cs new file mode 100644 index 0000000000..479d6ccea3 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/IntExpressionExtensions.cs @@ -0,0 +1,57 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Globalization; +using Microsoft.PowerFx; +using Microsoft.PowerFx.Types; + +namespace Microsoft.Bot.ObjectModel; + +/// +/// Extension methods for . +/// +internal static class IntExpressionExtensions +{ + /// + /// Evaluates the given using the provided . + /// + /// Expression to evaluate. + /// Recalc engine to use for evaluation. + /// The evaluated integer value, or null if the expression is null or cannot be evaluated. + internal static long? Eval(this IntExpression? expression, RecalcEngine? engine) + { + if (expression is null) + { + return null; + } + + if (expression.IsLiteral) + { + return expression.LiteralValue; + } + + if (engine is null) + { + return null; + } + + if (expression.IsExpression) + { + return (long)engine.Eval(expression.ExpressionText!).AsDouble(); + } + else if (expression.IsVariableReference) + { + var formulaValue = engine.Eval(expression.VariableReference!.VariableName); + if (formulaValue is NumberValue numberValue) + { + return (long)numberValue.Value; + } + + if (formulaValue is StringValue stringValue && int.TryParse(stringValue.Value, NumberStyles.Integer, CultureInfo.InvariantCulture, out int result)) + { + return result; + } + } + + return null; + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/McpServerToolApprovalModeExtensions.cs b/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/McpServerToolApprovalModeExtensions.cs new file mode 100644 index 0000000000..ee5632368b --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/McpServerToolApprovalModeExtensions.cs @@ -0,0 +1,30 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Microsoft.Extensions.AI; + +namespace Microsoft.Bot.ObjectModel; + +/// +/// Extension methods for . +/// +internal static class McpServerToolApprovalModeExtensions +{ + /// + /// Converts a to a . + /// + /// Instance of + internal static HostedMcpServerToolApprovalMode AsHostedMcpServerToolApprovalMode(this McpServerToolApprovalMode mode) + { + return mode switch + { + McpServerToolNeverRequireApprovalMode => HostedMcpServerToolApprovalMode.NeverRequire, + McpServerToolAlwaysRequireApprovalMode => HostedMcpServerToolApprovalMode.AlwaysRequire, + McpServerToolRequireSpecificApprovalMode specificMode => + HostedMcpServerToolApprovalMode.RequireSpecific( + specificMode?.AlwaysRequireApprovalToolNames?.LiteralValue ?? [], + specificMode?.NeverRequireApprovalToolNames?.LiteralValue ?? [] + ), + _ => HostedMcpServerToolApprovalMode.AlwaysRequire, + }; + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/McpServerToolExtensions.cs b/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/McpServerToolExtensions.cs new file mode 100644 index 0000000000..763e402625 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/McpServerToolExtensions.cs @@ -0,0 +1,35 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using Microsoft.Extensions.AI; +using Microsoft.Shared.Diagnostics; + +namespace Microsoft.Bot.ObjectModel; + +/// +/// Extension methods for . +/// +internal static class McpServerToolExtensions +{ + /// + /// Creates a from a . + /// + /// Instance of + internal static HostedMcpServerTool CreateHostedMcpTool(this McpServerTool tool) + { + Throw.IfNull(tool); + Throw.IfNull(tool.ServerName?.LiteralValue); + Throw.IfNull(tool.Connection); + + var connection = tool.Connection as AnonymousConnection ?? throw new ArgumentException("Only AnonymousConnection is supported for MCP Server Tool connections.", nameof(tool)); + var serverUrl = connection.Endpoint?.LiteralValue; + Throw.IfNullOrEmpty(serverUrl, nameof(connection.Endpoint)); + + return new HostedMcpServerTool(tool.ServerName.LiteralValue, serverUrl) + { + ServerDescription = tool.ServerDescription?.LiteralValue, + AllowedTools = tool.AllowedTools?.LiteralValue, + ApprovalMode = tool.ApprovalMode?.AsHostedMcpServerToolApprovalMode(), + }; + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/ModelOptionsExtensions.cs b/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/ModelOptionsExtensions.cs new file mode 100644 index 0000000000..7ad4d26a6b --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/ModelOptionsExtensions.cs @@ -0,0 +1,66 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Linq; +using Microsoft.Extensions.AI; +using Microsoft.Shared.Diagnostics; + +namespace Microsoft.Bot.ObjectModel; + +/// +/// Extension methods for . +/// +internal static class ModelOptionsExtensions +{ + /// + /// Converts the 'chatToolMode' property from a to a . + /// + /// Instance of + internal static ChatToolMode? AsChatToolMode(this ModelOptions modelOptions) + { + Throw.IfNull(modelOptions); + + var mode = modelOptions.ExtensionData?.GetPropertyOrNull(InitializablePropertyPath.Create("chatToolMode"))?.Value; + if (mode is null) + { + return null; + } + + return mode switch + { + "auto" => ChatToolMode.Auto, + "none" => ChatToolMode.None, + "require_any" => ChatToolMode.RequireAny, + _ => ChatToolMode.RequireSpecific(mode), + }; + } + + /// + /// Retrieves the 'additional_properties' property from a . + /// + /// Instance of + /// List of properties which should not be included in additional properties. + internal static AdditionalPropertiesDictionary? GetAdditionalProperties(this ModelOptions modelOptions, string[] excludedProperties) + { + Throw.IfNull(modelOptions); + + var options = modelOptions.ExtensionData; + if (options is null || options.Properties.Count == 0) + { + return null; + } + + var additionalProperties = options.Properties + .Where(kvp => !excludedProperties.Contains(kvp.Key)) + .ToDictionary( + kvp => kvp.Key, + kvp => kvp.Value?.ToObject()); + + if (additionalProperties is null || additionalProperties.Count == 0) + { + return null; + } + + return new AdditionalPropertiesDictionary(additionalProperties); + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/NumberExpressionExtensions.cs b/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/NumberExpressionExtensions.cs new file mode 100644 index 0000000000..cfa36185cc --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/NumberExpressionExtensions.cs @@ -0,0 +1,57 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Globalization; +using Microsoft.PowerFx; +using Microsoft.PowerFx.Types; + +namespace Microsoft.Bot.ObjectModel; + +/// +/// Extension methods for . +/// +internal static class NumberExpressionExtensions +{ + /// + /// Evaluates the given using the provided . + /// + /// Expression to evaluate. + /// Recalc engine to use for evaluation. + /// The evaluated number value, or null if the expression is null or cannot be evaluated. + internal static double? Eval(this NumberExpression? expression, RecalcEngine? engine) + { + if (expression is null) + { + return null; + } + + if (expression.IsLiteral) + { + return expression.LiteralValue; + } + + if (engine is null) + { + return null; + } + + if (expression.IsExpression) + { + return engine.Eval(expression.ExpressionText!).AsDouble(); + } + else if (expression.IsVariableReference) + { + var formulaValue = engine.Eval(expression.VariableReference!.VariableName); + if (formulaValue is NumberValue numberValue) + { + return numberValue.Value; + } + + if (formulaValue is StringValue stringValue && double.TryParse(stringValue.Value, NumberStyles.Float, CultureInfo.InvariantCulture, out double result)) + { + return result; + } + } + + return null; + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/PromptAgentExtensions.cs b/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/PromptAgentExtensions.cs new file mode 100644 index 0000000000..1597c0c54b --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/PromptAgentExtensions.cs @@ -0,0 +1,114 @@ +// Copyright (c) Microsoft. All rights reserved. +using System; +using System.Collections.Generic; +using System.Linq; +using Microsoft.Extensions.AI; +using Microsoft.PowerFx; +using Microsoft.Shared.Diagnostics; + +namespace Microsoft.Bot.ObjectModel; + +/// +/// Extension methods for . +/// +public static class PromptAgentExtensions +{ + /// + /// Retrieves the 'options' property from a as a instance. + /// + /// Instance of + /// Instance of + /// Instance of + public static ChatOptions? GetChatOptions(this GptComponentMetadata promptAgent, RecalcEngine? engine, IList? functions) + { + Throw.IfNull(promptAgent); + + var outputSchema = promptAgent.OutputType; + var modelOptions = promptAgent.Model?.Options; + + var tools = promptAgent.GetAITools(functions); + + if (modelOptions is null && tools is null) + { + return null; + } + + return new ChatOptions() + { + Instructions = promptAgent.Instructions?.ToTemplateString(), + Temperature = (float?)modelOptions?.Temperature?.Eval(engine), + MaxOutputTokens = (int?)modelOptions?.MaxOutputTokens?.Eval(engine), + TopP = (float?)modelOptions?.TopP?.Eval(engine), + TopK = (int?)modelOptions?.TopK?.Eval(engine), + FrequencyPenalty = (float?)modelOptions?.FrequencyPenalty?.Eval(engine), + PresencePenalty = (float?)modelOptions?.PresencePenalty?.Eval(engine), + Seed = modelOptions?.Seed?.Eval(engine), + ResponseFormat = outputSchema?.AsChatResponseFormat(), + ModelId = promptAgent.Model?.ModelNameHint, + StopSequences = modelOptions?.StopSequences, + AllowMultipleToolCalls = modelOptions?.AllowMultipleToolCalls?.Eval(engine), + ToolMode = modelOptions?.AsChatToolMode(), + Tools = tools, + AdditionalProperties = modelOptions?.GetAdditionalProperties(s_chatOptionProperties), + }; + } + + /// + /// Retrieves the 'tools' property from a . + /// + /// Instance of + /// Instance of + internal static List? GetAITools(this GptComponentMetadata promptAgent, IList? functions) + { + return promptAgent.Tools.Select(tool => + { + return tool switch + { + CodeInterpreterTool => ((CodeInterpreterTool)tool).AsCodeInterpreterTool(), + InvokeClientTaskAction => ((InvokeClientTaskAction)tool).CreateOrGetAITool(functions), + McpServerTool => ((McpServerTool)tool).CreateHostedMcpTool(), + FileSearchTool => ((FileSearchTool)tool).CreateFileSearchTool(), + WebSearchTool => ((WebSearchTool)tool).CreateWebSearchTool(), + _ => throw new NotSupportedException($"Unable to create tool definition because of unsupported tool type: {tool.Kind}, supported tool types are: {string.Join(",", s_validToolKinds)}"), + }; + }).ToList() ?? []; + } + + #region private + private const string CodeInterpreterKind = "codeInterpreter"; + private const string FileSearchKind = "fileSearch"; + private const string FunctionKind = "function"; + private const string WebSearchKind = "webSearch"; + private const string McpKind = "mcp"; + + private static readonly string[] s_validToolKinds = + [ + CodeInterpreterKind, + FileSearchKind, + FunctionKind, + WebSearchKind, + McpKind + ]; + + private static readonly string[] s_chatOptionProperties = + [ + "allowMultipleToolCalls", + "conversationId", + "chatToolMode", + "frequencyPenalty", + "additionalInstructions", + "maxOutputTokens", + "modelId", + "presencePenalty", + "responseFormat", + "seed", + "stopSequences", + "temperature", + "topK", + "topP", + "toolMode", + "tools", + ]; + + #endregion +} diff --git a/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/PropertyInfoExtensions.cs b/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/PropertyInfoExtensions.cs new file mode 100644 index 0000000000..a62fddec88 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/PropertyInfoExtensions.cs @@ -0,0 +1,96 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Collections.Generic; +using System.Text.Json; + +namespace Microsoft.Bot.ObjectModel; + +/// +/// Extension methods for . +/// +public static class PropertyInfoExtensions +{ + /// + /// Creates a of and + /// from an of and . + /// + /// A read-only dictionary of property names and their corresponding objects. + public static Dictionary AsObjectDictionary(this IReadOnlyDictionary properties) + { + var result = new Dictionary(); + + foreach (var property in properties) + { + result[property.Key] = BuildPropertySchema(property.Value); + } + + return result; + } + + #region private + private static Dictionary BuildPropertySchema(PropertyInfo propertyInfo) + { + var propertySchema = new Dictionary(); + + // Map the DataType to JSON schema type and add type-specific properties + switch (propertyInfo.Type) + { + case StringDataType: + propertySchema["type"] = "string"; + break; + case NumberDataType: + propertySchema["type"] = "number"; + break; + case BooleanDataType: + propertySchema["type"] = "boolean"; + break; + case DateTimeDataType: + propertySchema["type"] = "string"; + propertySchema["format"] = "date-time"; + break; + case DateDataType: + propertySchema["type"] = "string"; + propertySchema["format"] = "date"; + break; + case TimeDataType: + propertySchema["type"] = "string"; + propertySchema["format"] = "time"; + break; + case RecordDataType nestedRecordType: +#pragma warning disable IL2026, IL3050 + // For nested records, recursively build the schema + var nestedSchema = nestedRecordType.GetSchema(); + var nestedJson = JsonSerializer.Serialize(nestedSchema, ElementSerializer.CreateOptions()); + var nestedDict = JsonSerializer.Deserialize>(nestedJson, ElementSerializer.CreateOptions()); +#pragma warning restore IL2026, IL3050 + if (nestedDict != null) + { + return nestedDict; + } + propertySchema["type"] = "object"; + break; + case TableDataType tableType: + propertySchema["type"] = "array"; + // TableDataType has Properties like RecordDataType + propertySchema["items"] = new Dictionary + { + ["type"] = "object", + ["properties"] = AsObjectDictionary(tableType.Properties), + ["additionalProperties"] = false + }; + break; + default: + propertySchema["type"] = "string"; + break; + } + + // Add description if available + if (!string.IsNullOrEmpty(propertyInfo.Description)) + { + propertySchema["description"] = propertyInfo.Description; + } + + return propertySchema; + } + #endregion +} diff --git a/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/RecordDataTypeExtensions.cs b/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/RecordDataTypeExtensions.cs new file mode 100644 index 0000000000..b5c5793cab --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/RecordDataTypeExtensions.cs @@ -0,0 +1,77 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Collections.Generic; +using System.Text.Json; +using Microsoft.Extensions.AI; +using Microsoft.Shared.Diagnostics; + +namespace Microsoft.Bot.ObjectModel; + +/// +/// Extension methods for . +/// +public static class RecordDataTypeExtensions +{ + /// + /// Creates a from a . + /// + /// Instance of + internal static ChatResponseFormat? AsChatResponseFormat(this RecordDataType recordDataType) + { + Throw.IfNull(recordDataType); + + if (recordDataType.Properties.Count == 0) + { + return null; + } + + // TODO: Consider adding schemaName and schemaDescription parameters to this method. + return ChatResponseFormat.ForJsonSchema( + schema: recordDataType.GetSchema(), + schemaName: recordDataType.GetSchemaName(), + schemaDescription: recordDataType.GetSchemaDescription()); + } + + /// + /// Converts a to a . + /// + /// Instance of +#pragma warning disable IL2026 // Members annotated with 'RequiresUnreferencedCodeAttribute' require dynamic access otherwise can break functionality when trimming application code +#pragma warning disable IL3050 // Calling members annotated with 'RequiresDynamicCodeAttribute' may break functionality when AOT compiling. + public static JsonElement GetSchema(this RecordDataType recordDataType) + { + Throw.IfNull(recordDataType); + + var schemaObject = new Dictionary + { + ["type"] = "object", + ["properties"] = recordDataType.Properties.AsObjectDictionary(), + ["additionalProperties"] = false + }; + + var json = JsonSerializer.Serialize(schemaObject, ElementSerializer.CreateOptions()); + return JsonSerializer.Deserialize(json); + } +#pragma warning restore IL3050 // Calling members annotated with 'RequiresDynamicCodeAttribute' may break functionality when AOT compiling. +#pragma warning restore IL2026 // Members annotated with 'RequiresUnreferencedCodeAttribute' require dynamic access otherwise can break functionality when trimming application code + + /// + /// Retrieves the 'schemaName' property from a . + /// + private static string? GetSchemaName(this RecordDataType recordDataType) + { + Throw.IfNull(recordDataType); + + return recordDataType.ExtensionData?.GetPropertyOrNull(InitializablePropertyPath.Create("schemaName"))?.Value; + } + + /// + /// Retrieves the 'schemaDescription' property from a . + /// + private static string? GetSchemaDescription(this RecordDataType recordDataType) + { + Throw.IfNull(recordDataType); + + return recordDataType.ExtensionData?.GetPropertyOrNull(InitializablePropertyPath.Create("schemaDescription"))?.Value; + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/RecordDataValueExtensions.cs b/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/RecordDataValueExtensions.cs new file mode 100644 index 0000000000..6351b7badb --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/RecordDataValueExtensions.cs @@ -0,0 +1,107 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Collections.Generic; +using System.Linq; +using System.Text.Json; +using Microsoft.Shared.Diagnostics; + +namespace Microsoft.Bot.ObjectModel; + +/// +/// Extension methods for . +/// +public static class RecordDataValueExtensions +{ + /// + /// Retrieves a 'number' property from a + /// + /// Instance of + /// Path of the property to retrieve + public static decimal? GetNumber(this RecordDataValue recordData, string propertyPath) + { + Throw.IfNull(recordData); + + var numberValue = recordData.GetPropertyOrNull(InitializablePropertyPath.Create(propertyPath)); + return numberValue?.Value; + } + + /// + /// Retrieves a nullable boolean value from the specified property path within the given record data. + /// + /// Instance of + /// Path of the property to retrieve + public static bool? GetBoolean(this RecordDataValue recordData, string propertyPath) + { + Throw.IfNull(recordData); + + var booleanValue = recordData.GetPropertyOrNull(InitializablePropertyPath.Create(propertyPath)); + return booleanValue?.Value; + } + + /// + /// Converts a to a . + /// + /// Instance of + public static IReadOnlyDictionary ToDictionary(this RecordDataValue recordData) + { + Throw.IfNull(recordData); + + return recordData.Properties.ToDictionary( + kvp => kvp.Key, + kvp => kvp.Value?.ToString() ?? string.Empty + ); + } + + /// + /// Retrieves the 'schema' property from a . + /// + /// Instance of +#pragma warning disable IL2026 // Members annotated with 'RequiresUnreferencedCodeAttribute' require dynamic access otherwise can break functionality when trimming application code +#pragma warning disable IL3050 // Calling members annotated with 'RequiresDynamicCodeAttribute' may break functionality when AOT compiling. + public static JsonElement? GetSchema(this RecordDataValue recordData) + { + Throw.IfNull(recordData); + + try + { + var schemaStr = recordData.GetPropertyOrNull(InitializablePropertyPath.Create("json_schema.schema")); + if (schemaStr?.Value is not null) + { + return JsonSerializer.Deserialize(schemaStr.Value); + } + } + catch (InvalidCastException) + { + // Ignore and try next + } + + var responseFormRec = recordData.GetPropertyOrNull(InitializablePropertyPath.Create("json_schema.schema")); + if (responseFormRec is not null) + { + var json = JsonSerializer.Serialize(responseFormRec, ElementSerializer.CreateOptions()); + return JsonSerializer.Deserialize(json); + } + + return null; + } +#pragma warning restore IL3050 // Calling members annotated with 'RequiresDynamicCodeAttribute' may break functionality when AOT compiling. +#pragma warning restore IL2026 // Members annotated with 'RequiresUnreferencedCodeAttribute' require dynamic access otherwise can break functionality when trimming application code + + internal static object? ToObject(this DataValue? value) + { + if (value is null) + { + return null; + } + return value switch + { + StringDataValue s => s.Value, + NumberDataValue n => n.Value, + BooleanDataValue b => b.Value, + TableDataValue t => t.Values.Select(v => v.ToObject()).ToList(), + RecordDataValue r => r.Properties.ToDictionary(kvp => kvp.Key, kvp => kvp.Value?.ToObject()), + _ => throw new NotSupportedException($"Unsupported DataValue type: {value.GetType().FullName}"), + }; + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/StringExpressionExtensions.cs b/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/StringExpressionExtensions.cs new file mode 100644 index 0000000000..40c1b7c9c8 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/StringExpressionExtensions.cs @@ -0,0 +1,48 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Microsoft.PowerFx; +using Microsoft.PowerFx.Types; + +namespace Microsoft.Bot.ObjectModel; + +/// +/// Extension methods for . +/// +public static class StringExpressionExtensions +{ + /// + /// Evaluates the given using the provided . + /// + /// Expression to evaluate. + /// Recalc engine to use for evaluation. + /// The evaluated string value, or null if the expression is null or cannot be evaluated. + public static string? Eval(this StringExpression? expression, RecalcEngine? engine) + { + if (expression is null) + { + return null; + } + + if (expression.IsLiteral) + { + return expression.LiteralValue?.ToString(); + } + + if (engine is null) + { + return null; + } + + if (expression.IsExpression) + { + return engine.Eval(expression.ExpressionText!).ToString(); + } + else if (expression.IsVariableReference) + { + var stringValue = engine.Eval(expression.VariableReference!.VariableName) as StringValue; + return stringValue?.Value; + } + + return null; + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/WebSearchToolExtensions.cs b/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/WebSearchToolExtensions.cs new file mode 100644 index 0000000000..e6ee360308 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/WebSearchToolExtensions.cs @@ -0,0 +1,23 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Microsoft.Extensions.AI; +using Microsoft.Shared.Diagnostics; + +namespace Microsoft.Bot.ObjectModel; + +/// +/// Extension methods for . +/// +internal static class WebSearchToolExtensions +{ + /// + /// Create a from a . + /// + /// Instance of + internal static HostedWebSearchTool CreateWebSearchTool(this WebSearchTool tool) + { + Throw.IfNull(tool); + + return new HostedWebSearchTool(); + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/YamlAgentFactoryExtensions.cs b/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/YamlAgentFactoryExtensions.cs new file mode 100644 index 0000000000..1cc24055d9 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Declarative/Extensions/YamlAgentFactoryExtensions.cs @@ -0,0 +1,33 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Diagnostics.CodeAnalysis; +using System.Threading; +using System.Threading.Tasks; +using Microsoft.Shared.Diagnostics; + +namespace Microsoft.Agents.AI; + +/// +/// Extension methods for to support YAML based agent definitions. +/// +public static class YamlAgentFactoryExtensions +{ + /// + /// Create a from the given agent YAML. + /// + /// which will be used to create the agent. + /// Text string containing the YAML representation of an . + /// Optional cancellation token + [RequiresDynamicCode("Calls YamlDotNet.Serialization.DeserializerBuilder.DeserializerBuilder()")] + public static Task CreateFromYamlAsync(this PromptAgentFactory agentFactory, string agentYaml, CancellationToken cancellationToken = default) + { + Throw.IfNull(agentFactory); + Throw.IfNullOrEmpty(agentYaml); + + var agentDefinition = AgentBotElementYaml.FromYaml(agentYaml); + + return agentFactory.CreateAsync( + agentDefinition, + cancellationToken); + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Declarative/Microsoft.Agents.AI.Declarative.csproj b/dotnet/src/Microsoft.Agents.AI.Declarative/Microsoft.Agents.AI.Declarative.csproj new file mode 100644 index 0000000000..306ba27e97 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Declarative/Microsoft.Agents.AI.Declarative.csproj @@ -0,0 +1,45 @@ + + + + preview + $(NoWarn);MEAI001 + false + + + + true + true + true + + + + + + + Microsoft Agent Framework Declarative Agents + Provides Microsoft Agent Framework support for declarative agents. + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/dotnet/src/Microsoft.Agents.AI.Declarative/PromptAgentFactory.cs b/dotnet/src/Microsoft.Agents.AI.Declarative/PromptAgentFactory.cs new file mode 100644 index 0000000000..cb277b06da --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Declarative/PromptAgentFactory.cs @@ -0,0 +1,63 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Threading; +using System.Threading.Tasks; +using Microsoft.Bot.ObjectModel; +using Microsoft.Extensions.Configuration; +using Microsoft.PowerFx; +using Microsoft.Shared.Diagnostics; + +namespace Microsoft.Agents.AI; + +/// +/// Represents a factory for creating instances. +/// +public abstract class PromptAgentFactory +{ + /// + /// Initializes a new instance of the class. + /// + /// Optional , if none is provided a default instance will be created. + /// Optional configuration to be added as variables to the . + protected PromptAgentFactory(RecalcEngine? engine = null, IConfiguration? configuration = null) + { + this.Engine = engine ?? new RecalcEngine(); + + if (configuration is not null) + { + foreach (var kvp in configuration.AsEnumerable()) + { + this.Engine.UpdateVariable(kvp.Key, kvp.Value ?? string.Empty); + } + } + } + + /// + /// Gets the Power Fx recalculation engine used to evaluate expressions in agent definitions. + /// This engine is configured with variables from the provided during construction. + /// + protected RecalcEngine Engine { get; } + + /// + /// Create a from the specified . + /// + /// Definition of the agent to create. + /// Optional cancellation token. + /// The created , if null the agent type is not supported. + public async Task CreateAsync(GptComponentMetadata promptAgent, CancellationToken cancellationToken = default) + { + Throw.IfNull(promptAgent); + + var agent = await this.TryCreateAsync(promptAgent, cancellationToken).ConfigureAwait(false); + return agent ?? throw new NotSupportedException($"Agent type {promptAgent.Kind} is not supported."); + } + + /// + /// Tries to create a from the specified . + /// + /// Definition of the agent to create. + /// Optional cancellation token. + /// The created , if null the agent type is not supported. + public abstract Task TryCreateAsync(GptComponentMetadata promptAgent, CancellationToken cancellationToken = default); +} diff --git a/dotnet/src/Microsoft.Agents.AI.DevUI/DevUIExtensions.cs b/dotnet/src/Microsoft.Agents.AI.DevUI/DevUIExtensions.cs index 4a85de121a..8d5159cab7 100644 --- a/dotnet/src/Microsoft.Agents.AI.DevUI/DevUIExtensions.cs +++ b/dotnet/src/Microsoft.Agents.AI.DevUI/DevUIExtensions.cs @@ -9,32 +9,31 @@ namespace Microsoft.Agents.AI.DevUI; /// public static class DevUIExtensions { - /// - /// Adds the necessary services for the DevUI to the application builder. - /// - public static IHostApplicationBuilder AddDevUI(this IHostApplicationBuilder builder) - { - ArgumentNullException.ThrowIfNull(builder); - builder.Services.AddOpenAIConversations(); - builder.Services.AddOpenAIResponses(); - - return builder; - } - /// /// Maps an endpoint that serves the DevUI from the '/devui' path. /// + /// + /// DevUI requires the OpenAI Responses and Conversations services to be registered with + /// and + /// , + /// and the corresponding endpoints to be mapped using + /// and + /// . + /// /// The to add the endpoint to. /// A that can be used to add authorization or other endpoint configuration. + /// + /// + /// + /// /// Thrown when is null. public static IEndpointConventionBuilder MapDevUI( this IEndpointRouteBuilder endpoints) { var group = endpoints.MapGroup(""); group.MapDevUI(pattern: "/devui"); + group.MapMeta(); group.MapEntities(); - group.MapOpenAIConversations(); - group.MapOpenAIResponses(); return group; } diff --git a/dotnet/src/Microsoft.Agents.AI.DevUI/DevUIMiddleware.cs b/dotnet/src/Microsoft.Agents.AI.DevUI/DevUIMiddleware.cs index fc6dd512ec..ac585ad39a 100644 --- a/dotnet/src/Microsoft.Agents.AI.DevUI/DevUIMiddleware.cs +++ b/dotnet/src/Microsoft.Agents.AI.DevUI/DevUIMiddleware.cs @@ -4,6 +4,7 @@ using System.IO.Compression; using System.Reflection; using System.Security.Cryptography; +using System.Text.RegularExpressions; using Microsoft.AspNetCore.StaticFiles; using Microsoft.Extensions.Primitives; using Microsoft.Net.Http.Headers; @@ -13,8 +14,11 @@ namespace Microsoft.Agents.AI.DevUI; /// /// Handler that serves embedded DevUI resource files from the 'resources' directory. /// -internal sealed class DevUIMiddleware +internal sealed partial class DevUIMiddleware { + [GeneratedRegex(@"[\r\n]+")] + private static partial Regex NewlineRegex(); + private const string GZipEncodingValue = "gzip"; private static readonly StringValues s_gzipEncodingHeader = new(GZipEncodingValue); private static readonly Assembly s_assembly = typeof(DevUIMiddleware).Assembly; @@ -70,15 +74,20 @@ public async Task HandleRequestAsync(HttpContext context) // This ensures relative URLs in the HTML work correctly if (string.Equals(path, this._basePath, StringComparison.OrdinalIgnoreCase) && !path.EndsWith('/')) { - var redirectUrl = $"{path}/"; + var redirectUrl = this._basePath + "/"; if (context.Request.QueryString.HasValue) { redirectUrl += context.Request.QueryString.Value; } context.Response.StatusCode = StatusCodes.Status301MovedPermanently; - context.Response.Headers.Location = redirectUrl; - this._logger.LogDebug("Redirecting {OriginalPath} to {RedirectUrl}", path, redirectUrl); + context.Response.Headers.Location = redirectUrl; // CodeQL [SM04598] justification: The redirect URL is constructed from a server-configured base path (_basePath), not user input. The query string is only appended as parameters and cannot change the redirect destination since this is a relative URL. + + if (this._logger.IsEnabled(LogLevel.Debug)) + { + this._logger.LogDebug("Redirecting {OriginalPath} to {RedirectUrl}", NewlineRegex().Replace(path, ""), NewlineRegex().Replace(redirectUrl, "")); + } + return; } @@ -118,7 +127,11 @@ private async Task TryServeResourceAsync(HttpContext context, string resou { if (!this._resourceCache.TryGetValue(resourcePath.Replace('.', '/'), out var cacheEntry)) { - this._logger.LogDebug("Embedded resource not found: {ResourcePath}", resourcePath); + if (this._logger.IsEnabled(LogLevel.Debug)) + { + this._logger.LogDebug("Embedded resource not found: {ResourcePath}", resourcePath); + } + return false; } @@ -128,7 +141,12 @@ private async Task TryServeResourceAsync(HttpContext context, string resou if (context.Request.Headers.IfNoneMatch == cacheEntry.ETag) { response.StatusCode = StatusCodes.Status304NotModified; - this._logger.LogDebug("Resource not modified (304): {ResourcePath}", resourcePath); + + if (this._logger.IsEnabled(LogLevel.Debug)) + { + this._logger.LogDebug("Resource not modified (304): {ResourcePath}", resourcePath); + } + return true; } @@ -156,12 +174,20 @@ private async Task TryServeResourceAsync(HttpContext context, string resou await response.Body.WriteAsync(content, context.RequestAborted).ConfigureAwait(false); - this._logger.LogDebug("Served embedded resource: {ResourcePath} (compressed: {Compressed})", resourcePath, serveCompressed); + if (this._logger.IsEnabled(LogLevel.Debug)) + { + this._logger.LogDebug("Served embedded resource: {ResourcePath} (compressed: {Compressed})", resourcePath, serveCompressed); + } + return true; } catch (Exception ex) { - this._logger.LogError(ex, "Error serving embedded resource: {ResourcePath}", resourcePath); + if (this._logger.IsEnabled(LogLevel.Error)) + { + this._logger.LogError(ex, "Error serving embedded resource: {ResourcePath}", resourcePath); + } + return false; } } diff --git a/dotnet/src/Microsoft.Agents.AI.DevUI/Entities/EntitiesJsonContext.cs b/dotnet/src/Microsoft.Agents.AI.DevUI/Entities/EntitiesJsonContext.cs index fc8bbe3864..09b95769a9 100644 --- a/dotnet/src/Microsoft.Agents.AI.DevUI/Entities/EntitiesJsonContext.cs +++ b/dotnet/src/Microsoft.Agents.AI.DevUI/Entities/EntitiesJsonContext.cs @@ -15,10 +15,16 @@ namespace Microsoft.Agents.AI.DevUI.Entities; DefaultIgnoreCondition = JsonIgnoreCondition.WhenWritingNull)] [JsonSerializable(typeof(EntityInfo))] [JsonSerializable(typeof(DiscoveryResponse))] +[JsonSerializable(typeof(MetaResponse))] [JsonSerializable(typeof(EnvVarRequirement))] [JsonSerializable(typeof(List))] -[JsonSerializable(typeof(List))] +[JsonSerializable(typeof(List>))] +[JsonSerializable(typeof(List>))] [JsonSerializable(typeof(Dictionary))] +[JsonSerializable(typeof(Dictionary>))] +[JsonSerializable(typeof(Dictionary))] [JsonSerializable(typeof(JsonElement))] +[JsonSerializable(typeof(string))] +[JsonSerializable(typeof(int))] [ExcludeFromCodeCoverage] internal sealed partial class EntitiesJsonContext : JsonSerializerContext; diff --git a/dotnet/src/Microsoft.Agents.AI.DevUI/Entities/EntityInfo.cs b/dotnet/src/Microsoft.Agents.AI.DevUI/Entities/EntityInfo.cs index 8b5e4e5492..7b711b36c2 100644 --- a/dotnet/src/Microsoft.Agents.AI.DevUI/Entities/EntityInfo.cs +++ b/dotnet/src/Microsoft.Agents.AI.DevUI/Entities/EntityInfo.cs @@ -36,16 +36,16 @@ internal sealed record EntityInfo( string Name, [property: JsonPropertyName("description")] - string? Description = null, + string? Description, [property: JsonPropertyName("framework")] - string Framework = "dotnet", + string Framework, [property: JsonPropertyName("tools")] - List? Tools = null, + List Tools, [property: JsonPropertyName("metadata")] - Dictionary? Metadata = null + Dictionary Metadata ) { [JsonPropertyName("source")] @@ -54,6 +54,32 @@ internal sealed record EntityInfo( [JsonPropertyName("original_url")] public string? OriginalUrl { get; init; } + // Deployment support + [JsonPropertyName("deployment_supported")] + public bool DeploymentSupported { get; init; } + + [JsonPropertyName("deployment_reason")] + public string? DeploymentReason { get; init; } + + // Agent-specific fields + [JsonPropertyName("instructions")] + public string? Instructions { get; init; } + + [JsonPropertyName("model_id")] + public string? ModelId { get; init; } + + [JsonPropertyName("chat_client_type")] + public string? ChatClientType { get; init; } + + [JsonPropertyName("context_providers")] + public List? ContextProviders { get; init; } + + [JsonPropertyName("middleware")] + public List? Middleware { get; init; } + + [JsonPropertyName("module_path")] + public string? ModulePath { get; init; } + // Workflow-specific fields [JsonPropertyName("required_env_vars")] public List? RequiredEnvVars { get; init; } diff --git a/dotnet/src/Microsoft.Agents.AI.DevUI/Entities/MetaResponse.cs b/dotnet/src/Microsoft.Agents.AI.DevUI/Entities/MetaResponse.cs new file mode 100644 index 0000000000..df717c6952 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DevUI/Entities/MetaResponse.cs @@ -0,0 +1,66 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +namespace Microsoft.Agents.AI.DevUI.Entities; + +/// +/// Server metadata response for the /meta endpoint. +/// Provides information about the DevUI server configuration, capabilities, and requirements. +/// +/// +/// This response is used by the frontend to: +/// - Determine the UI mode (developer vs user interface) +/// - Check server capabilities (tracing, OpenAI proxy support) +/// - Verify authentication requirements +/// - Display framework and version information +/// +internal sealed record MetaResponse +{ + /// + /// Gets the UI interface mode. + /// "developer" shows debug tools and advanced features, "user" shows a simplified interface. + /// + [JsonPropertyName("ui_mode")] + public string UiMode { get; init; } = "developer"; + + /// + /// Gets the DevUI version string. + /// + [JsonPropertyName("version")] + public string Version { get; init; } = "0.1.0"; + + /// + /// Gets the backend framework identifier. + /// Always "agent_framework" for Agent Framework implementations. + /// + [JsonPropertyName("framework")] + public string Framework { get; init; } = "agent_framework"; + + /// + /// Gets the backend runtime/language. + /// "dotnet" for .NET implementations, "python" for Python implementations. + /// Used by frontend for deployment guides and feature availability. + /// + [JsonPropertyName("runtime")] + public string Runtime { get; init; } = "dotnet"; + + /// + /// Gets the server capabilities dictionary. + /// Key-value pairs indicating which optional features are enabled. + /// + /// + /// Standard capability keys: + /// - "tracing": Whether trace events are emitted for debugging + /// - "openai_proxy": Whether the server can proxy requests to OpenAI + /// + [JsonPropertyName("capabilities")] + public Dictionary Capabilities { get; init; } = []; + + /// + /// Gets a value indicating whether Bearer token authentication is required for API access. + /// When true, clients must include "Authorization: Bearer {token}" header in requests. + /// + [JsonPropertyName("auth_required")] + public bool AuthRequired { get; init; } +} diff --git a/dotnet/src/Microsoft.Agents.AI.DevUI/Entities/WorkflowSerializationExtensions.cs b/dotnet/src/Microsoft.Agents.AI.DevUI/Entities/WorkflowSerializationExtensions.cs index 81ce6182d1..44fc8b1eb4 100644 --- a/dotnet/src/Microsoft.Agents.AI.DevUI/Entities/WorkflowSerializationExtensions.cs +++ b/dotnet/src/Microsoft.Agents.AI.DevUI/Entities/WorkflowSerializationExtensions.cs @@ -1,5 +1,7 @@ // Copyright (c) Microsoft. All rights reserved. +using System.Text.Json; +using System.Text.Json.Serialization.Metadata; using Microsoft.Agents.AI.Workflows; using Microsoft.Agents.AI.Workflows.Checkpointing; @@ -17,31 +19,37 @@ internal static class WorkflowSerializationExtensions /// Converts a workflow to a dictionary representation compatible with DevUI frontend. /// This matches the Python workflow.to_dict() format expected by the UI. /// - public static Dictionary ToDevUIDict(this Workflow workflow) + /// The workflow to convert. + /// A dictionary with string keys and JsonElement values containing the workflow data. + public static Dictionary ToDevUIDict(this Workflow workflow) { - var result = new Dictionary + var result = new Dictionary { - ["id"] = workflow.Name ?? Guid.NewGuid().ToString(), - ["start_executor_id"] = workflow.StartExecutorId, - ["max_iterations"] = MaxIterationsDefault + ["id"] = Serialize(workflow.Name ?? Guid.NewGuid().ToString(), EntitiesJsonContext.Default.String), + ["start_executor_id"] = Serialize(workflow.StartExecutorId, EntitiesJsonContext.Default.String), + ["max_iterations"] = Serialize(MaxIterationsDefault, EntitiesJsonContext.Default.Int32) }; // Add optional fields if (!string.IsNullOrEmpty(workflow.Name)) { - result["name"] = workflow.Name; + result["name"] = Serialize(workflow.Name, EntitiesJsonContext.Default.String); } if (!string.IsNullOrEmpty(workflow.Description)) { - result["description"] = workflow.Description; + result["description"] = Serialize(workflow.Description, EntitiesJsonContext.Default.String); } // Convert executors to Python-compatible format - result["executors"] = ConvertExecutorsToDict(workflow); + result["executors"] = Serialize( + ConvertExecutorsToDict(workflow), + EntitiesJsonContext.Default.DictionaryStringDictionaryStringString); // Convert edges to edge_groups format - result["edge_groups"] = ConvertEdgesToEdgeGroups(workflow); + result["edge_groups"] = Serialize( + ConvertEdgesToEdgeGroups(workflow), + EntitiesJsonContext.Default.ListDictionaryStringJsonElement); return result; } @@ -49,9 +57,9 @@ public static Dictionary ToDevUIDict(this Workflow workflow) /// /// Converts workflow executors to a dictionary format compatible with Python /// - private static Dictionary ConvertExecutorsToDict(Workflow workflow) + private static Dictionary> ConvertExecutorsToDict(Workflow workflow) { - var executors = new Dictionary(); + var executors = new Dictionary>(); // Extract executor IDs from edges and start executor // (Registrations is internal, so we infer executors from the graph structure) @@ -73,7 +81,7 @@ private static Dictionary ConvertExecutorsToDict(Workflow workfl // Create executor entries (we can't access internal Registrations for type info) foreach (var executorId in executorIds) { - executors[executorId] = new Dictionary + executors[executorId] = new Dictionary { ["id"] = executorId, ["type"] = "Executor" @@ -86,9 +94,9 @@ private static Dictionary ConvertExecutorsToDict(Workflow workfl /// /// Converts workflow edges to edge_groups format expected by the UI /// - private static List ConvertEdgesToEdgeGroups(Workflow workflow) + private static List> ConvertEdgesToEdgeGroups(Workflow workflow) { - var edgeGroups = new List(); + var edgeGroups = new List>(); var edgeGroupId = 0; // Get edges using the public ReflectEdges method @@ -101,13 +109,13 @@ private static List ConvertEdgesToEdgeGroups(Workflow workflow) if (edgeInfo is DirectEdgeInfo directEdge) { // Single edge group for direct edges - var edges = new List(); + var edges = new List>(); foreach (var source in directEdge.Connection.SourceIds) { foreach (var sink in directEdge.Connection.SinkIds) { - var edge = new Dictionary + var edge = new Dictionary { ["source_id"] = source, ["target_id"] = sink @@ -123,23 +131,25 @@ private static List ConvertEdgesToEdgeGroups(Workflow workflow) } } - edgeGroups.Add(new Dictionary + var edgeGroup = new Dictionary { - ["id"] = $"edge_group_{edgeGroupId++}", - ["type"] = "SingleEdgeGroup", - ["edges"] = edges - }); + ["id"] = Serialize($"edge_group_{edgeGroupId++}", EntitiesJsonContext.Default.String), + ["type"] = Serialize("SingleEdgeGroup", EntitiesJsonContext.Default.String), + ["edges"] = Serialize(edges, EntitiesJsonContext.Default.ListDictionaryStringString) + }; + + edgeGroups.Add(edgeGroup); } else if (edgeInfo is FanOutEdgeInfo fanOutEdge) { // FanOut edge group - var edges = new List(); + var edges = new List>(); foreach (var source in fanOutEdge.Connection.SourceIds) { foreach (var sink in fanOutEdge.Connection.SinkIds) { - edges.Add(new Dictionary + edges.Add(new Dictionary { ["source_id"] = source, ["target_id"] = sink @@ -147,16 +157,16 @@ private static List ConvertEdgesToEdgeGroups(Workflow workflow) } } - var fanOutGroup = new Dictionary + var fanOutGroup = new Dictionary { - ["id"] = $"edge_group_{edgeGroupId++}", - ["type"] = "FanOutEdgeGroup", - ["edges"] = edges + ["id"] = Serialize($"edge_group_{edgeGroupId++}", EntitiesJsonContext.Default.String), + ["type"] = Serialize("FanOutEdgeGroup", EntitiesJsonContext.Default.String), + ["edges"] = Serialize(edges, EntitiesJsonContext.Default.ListDictionaryStringString) }; if (fanOutEdge.HasAssigner) { - fanOutGroup["selection_func_name"] = "selector"; + fanOutGroup["selection_func_name"] = Serialize("selector", EntitiesJsonContext.Default.String); } edgeGroups.Add(fanOutGroup); @@ -164,13 +174,13 @@ private static List ConvertEdgesToEdgeGroups(Workflow workflow) else if (edgeInfo is FanInEdgeInfo fanInEdge) { // FanIn edge group - var edges = new List(); + var edges = new List>(); foreach (var source in fanInEdge.Connection.SourceIds) { foreach (var sink in fanInEdge.Connection.SinkIds) { - edges.Add(new Dictionary + edges.Add(new Dictionary { ["source_id"] = source, ["target_id"] = sink @@ -178,16 +188,20 @@ private static List ConvertEdgesToEdgeGroups(Workflow workflow) } } - edgeGroups.Add(new Dictionary + var edgeGroup = new Dictionary { - ["id"] = $"edge_group_{edgeGroupId++}", - ["type"] = "FanInEdgeGroup", - ["edges"] = edges - }); + ["id"] = Serialize($"edge_group_{edgeGroupId++}", EntitiesJsonContext.Default.String), + ["type"] = Serialize("FanInEdgeGroup", EntitiesJsonContext.Default.String), + ["edges"] = Serialize(edges, EntitiesJsonContext.Default.ListDictionaryStringString) + }; + + edgeGroups.Add(edgeGroup); } } } return edgeGroups; } + + private static JsonElement Serialize(T value, JsonTypeInfo typeInfo) => JsonSerializer.SerializeToElement(value, typeInfo); } diff --git a/dotnet/src/Microsoft.Agents.AI.DevUI/EntitiesApiExtensions.cs b/dotnet/src/Microsoft.Agents.AI.DevUI/EntitiesApiExtensions.cs index 716dab8542..3271b40853 100644 --- a/dotnet/src/Microsoft.Agents.AI.DevUI/EntitiesApiExtensions.cs +++ b/dotnet/src/Microsoft.Agents.AI.DevUI/EntitiesApiExtensions.cs @@ -1,9 +1,9 @@ // Copyright (c) Microsoft. All rights reserved. using System.Text.Json; - using Microsoft.Agents.AI.DevUI.Entities; -using Microsoft.Agents.AI.Hosting; +using Microsoft.Agents.AI.Workflows; +using Microsoft.Extensions.AI; namespace Microsoft.Agents.AI.DevUI; @@ -24,21 +24,26 @@ internal static class EntitiesApiExtensions /// GET /v1/entities/{entityId}/info - Get detailed information about a specific entity /// /// The endpoints are compatible with the Python DevUI frontend and automatically discover entities - /// from the registered and services. + /// from the registered agents and workflows in the dependency injection container. /// public static IEndpointConventionBuilder MapEntities(this IEndpointRouteBuilder endpoints) { + var registeredAIAgents = GetRegisteredEntities(endpoints.ServiceProvider); + var registeredWorkflows = GetRegisteredEntities(endpoints.ServiceProvider); + var group = endpoints.MapGroup("/v1/entities") .WithTags("Entities"); // List all entities - group.MapGet("", ListEntitiesAsync) + group.MapGet("", (CancellationToken cancellationToken) + => ListEntitiesAsync(registeredAIAgents, registeredWorkflows, cancellationToken)) .WithName("ListEntities") .WithSummary("List all registered entities (agents and workflows)") .Produces(StatusCodes.Status200OK, contentType: "application/json"); // Get detailed entity information - group.MapGet("{entityId}/info", GetEntityInfoAsync) + group.MapGet("{entityId}/info", (string entityId, string? type, CancellationToken cancellationToken) + => GetEntityInfoAsync(entityId, type, registeredAIAgents, registeredWorkflows, cancellationToken)) .WithName("GetEntityInfo") .WithSummary("Get detailed information about a specific entity") .Produces(StatusCodes.Status200OK, contentType: "application/json") @@ -48,87 +53,27 @@ public static IEndpointConventionBuilder MapEntities(this IEndpointRouteBuilder } private static async Task ListEntitiesAsync( - AgentCatalog? agentCatalog, - WorkflowCatalog? workflowCatalog, + IEnumerable agents, + IEnumerable workflows, CancellationToken cancellationToken) { try { - var entities = new List(); + var entities = new Dictionary(); - // Discover agents from the agent catalog - if (agentCatalog is not null) + // Discover agents + foreach (var agentInfo in DiscoverAgents(agents, entityIdFilter: null)) { - await foreach (var agent in agentCatalog.GetAgentsAsync(cancellationToken).ConfigureAwait(false)) - { - if (agent.GetType().Name == "WorkflowHostAgent") - { - // HACK: ignore WorkflowHostAgent instances as they are just wrappers around workflows, - // and workflows are handled below. - continue; - } - - entities.Add(new EntityInfo( - Id: agent.Name ?? agent.Id, - Type: "agent", - Name: agent.Name ?? agent.Id, - Description: agent.Description, - Framework: "agent-framework", - Tools: null, - Metadata: [] - ) - { - Source = "in_memory" - }); - } + entities[agentInfo.Id] = agentInfo; } - // Discover workflows from the workflow catalog - if (workflowCatalog is not null) + // Discover workflows + foreach (var workflowInfo in DiscoverWorkflows(workflows, entityIdFilter: null)) { - await foreach (var workflow in workflowCatalog.GetWorkflowsAsync(cancellationToken).ConfigureAwait(false)) - { - // Extract executor IDs from the workflow structure - var executorIds = new HashSet { workflow.StartExecutorId }; - var reflectedEdges = workflow.ReflectEdges(); - foreach (var (sourceId, edgeSet) in reflectedEdges) - { - executorIds.Add(sourceId); - foreach (var edge in edgeSet) - { - foreach (var sinkId in edge.Connection.SinkIds) - { - executorIds.Add(sinkId); - } - } - } - - // Create a default input schema (string type) - var defaultInputSchema = new Dictionary - { - ["type"] = "string" - }; - - entities.Add(new EntityInfo( - Id: workflow.Name ?? workflow.StartExecutorId, - Type: "workflow", - Name: workflow.Name ?? workflow.StartExecutorId, - Description: workflow.Description, - Framework: "agent-framework", - Tools: [.. executorIds], - Metadata: [] - ) - { - Source = "in_memory", - WorkflowDump = JsonSerializer.SerializeToElement(workflow.ToDevUIDict()), - InputSchema = JsonSerializer.SerializeToElement(defaultInputSchema), - InputTypeName = "string", - StartExecutorId = workflow.StartExecutorId - }); - } + entities[workflowInfo.Id] = workflowInfo; } - return Results.Json(new DiscoveryResponse(entities), EntitiesJsonContext.Default.DiscoveryResponse); + return Results.Json(new DiscoveryResponse([.. entities.Values.OrderBy(e => e.Id)]), EntitiesJsonContext.Default.DiscoveryResponse); } catch (Exception ex) { @@ -141,93 +86,26 @@ private static async Task ListEntitiesAsync( private static async Task GetEntityInfoAsync( string entityId, - AgentCatalog? agentCatalog, - WorkflowCatalog? workflowCatalog, + string? type, + IEnumerable agents, + IEnumerable workflows, CancellationToken cancellationToken) { try { - // Try to find the entity among discovered agents - if (agentCatalog is not null) + if (type is null || string.Equals(type, "workflow", StringComparison.OrdinalIgnoreCase)) { - await foreach (var agent in agentCatalog.GetAgentsAsync(cancellationToken).ConfigureAwait(false)) + foreach (var workflowInfo in DiscoverWorkflows(workflows, entityId)) { - if (agent.GetType().Name == "WorkflowHostAgent") - { - // HACK: ignore WorkflowHostAgent instances as they are just wrappers around workflows, - // and workflows are handled below. - continue; - } - - if (string.Equals(agent.Name, entityId, StringComparison.OrdinalIgnoreCase) || - string.Equals(agent.Id, entityId, StringComparison.OrdinalIgnoreCase)) - { - var entityInfo = new EntityInfo( - Id: agent.Name ?? agent.Id, - Type: "agent", - Name: agent.Name ?? agent.Id, - Description: agent.Description, - Framework: "agent-framework", - Tools: null, - Metadata: [] - ) - { - Source = "in_memory" - }; - - return Results.Json(entityInfo, EntitiesJsonContext.Default.EntityInfo); - } + return Results.Json(workflowInfo, EntitiesJsonContext.Default.EntityInfo); } } - // Try to find the entity among discovered workflows - if (workflowCatalog is not null) + if (type is null || string.Equals(type, "agent", StringComparison.OrdinalIgnoreCase)) { - await foreach (var workflow in workflowCatalog.GetWorkflowsAsync(cancellationToken).ConfigureAwait(false)) + foreach (var agentInfo in DiscoverAgents(agents, entityId)) { - var workflowId = workflow.Name ?? workflow.StartExecutorId; - if (string.Equals(workflowId, entityId, StringComparison.OrdinalIgnoreCase)) - { - // Extract executor IDs from the workflow structure - var executorIds = new HashSet { workflow.StartExecutorId }; - var reflectedEdges = workflow.ReflectEdges(); - foreach (var (sourceId, edgeSet) in reflectedEdges) - { - executorIds.Add(sourceId); - foreach (var edge in edgeSet) - { - foreach (var sinkId in edge.Connection.SinkIds) - { - executorIds.Add(sinkId); - } - } - } - - // Create a default input schema (string type) - var defaultInputSchema = new Dictionary - { - ["type"] = "string" - }; - - var entityInfo = new EntityInfo( - Id: workflowId, - Type: "workflow", - Name: workflow.Name ?? workflow.StartExecutorId, - Description: workflow.Description, - Framework: "agent-framework", - Tools: [.. executorIds], - Metadata: [] - ) - { - Source = "in_memory", - WorkflowDump = JsonSerializer.SerializeToElement(workflow.ToDevUIDict()), - InputSchema = JsonSerializer.SerializeToElement(defaultInputSchema), - InputTypeName = "Input", - StartExecutorId = workflow.StartExecutorId - }; - - return Results.Json(entityInfo, EntitiesJsonContext.Default.EntityInfo); - } + return Results.Json(agentInfo, EntitiesJsonContext.Default.EntityInfo); } } @@ -241,4 +119,185 @@ private static async Task GetEntityInfoAsync( title: "Error getting entity info"); } } + + private static IEnumerable DiscoverAgents(IEnumerable agents, string? entityIdFilter) + { + foreach (var agent in agents) + { + // If filtering by entity ID, skip non-matching agents + if (entityIdFilter is not null && + !string.Equals(agent.Name, entityIdFilter, StringComparison.OrdinalIgnoreCase) && + !string.Equals(agent.Id, entityIdFilter, StringComparison.OrdinalIgnoreCase)) + { + continue; + } + + yield return CreateAgentEntityInfo(agent); + + // If we found the entity we're looking for, we're done + if (entityIdFilter is not null) + { + yield break; + } + } + } + + private static IEnumerable DiscoverWorkflows(IEnumerable workflows, string? entityIdFilter) + { + foreach (var workflow in workflows) + { + var workflowId = workflow.Name ?? workflow.StartExecutorId; + + // If filtering by entity ID, skip non-matching workflows + if (entityIdFilter is not null && !string.Equals(workflowId, entityIdFilter, StringComparison.OrdinalIgnoreCase)) + { + continue; + } + + yield return CreateWorkflowEntityInfo(workflow); + + // If we found the entity we're looking for, we're done + if (entityIdFilter is not null) + { + yield break; + } + } + } + + private static EntityInfo CreateAgentEntityInfo(AIAgent agent) + { + var entityId = agent.Name ?? agent.Id; + + // Extract tools and other metadata using GetService + List tools = []; + var metadata = new Dictionary(); + + // Try to get ChatOptions from the agent which may contain tools + if (agent.GetService() is { Tools: { Count: > 0 } agentTools }) + { + tools = agentTools + .Where(tool => !string.IsNullOrWhiteSpace(tool.Name)) + .Select(tool => tool.Name!) + .Distinct() + .ToList(); + } + + // Extract agent-specific fields (top-level properties for compatibility with Python) + string? instructions = null; + string? modelId = null; + string? chatClientType = null; + + // Get instructions from ChatClientAgent + if (agent is ChatClientAgent chatAgent && !string.IsNullOrWhiteSpace(chatAgent.Instructions)) + { + instructions = chatAgent.Instructions; + } + + // Get IChatClient to extract metadata + IChatClient? chatClient = agent.GetService(); + if (chatClient != null) + { + // Get chat client type + chatClientType = chatClient.GetType().Name; + + // Get model ID from ChatClientMetadata + if (chatClient.GetService() is { } chatClientMetadata) + { + modelId = chatClientMetadata.DefaultModelId; + + // Add additional metadata for compatibility + if (!string.IsNullOrWhiteSpace(chatClientMetadata.ProviderName)) + { + metadata["chat_client_provider"] = JsonSerializer.SerializeToElement(chatClientMetadata.ProviderName, EntitiesJsonContext.Default.String); + } + + if (chatClientMetadata.ProviderUri is not null) + { + metadata["provider_uri"] = JsonSerializer.SerializeToElement(chatClientMetadata.ProviderUri.ToString(), EntitiesJsonContext.Default.String); + } + } + } + + // Add provider name from AIAgentMetadata if available + if (agent.GetService() is { } agentMetadata && !string.IsNullOrWhiteSpace(agentMetadata.ProviderName)) + { + metadata["provider_name"] = JsonSerializer.SerializeToElement(agentMetadata.ProviderName, EntitiesJsonContext.Default.String); + } + + // Add agent type information to metadata (in addition to chat_client_type) + var agentTypeName = agent.GetType().Name; + metadata["agent_type"] = JsonSerializer.SerializeToElement(agentTypeName, EntitiesJsonContext.Default.String); + + return new EntityInfo( + Id: entityId, + Type: "agent", + Name: agent.DisplayName, + Description: agent.Description, + Framework: "agent_framework", + Tools: tools, + Metadata: metadata + ) + { + Source = "in_memory", + Instructions = instructions, + ModelId = modelId, + ChatClientType = chatClientType, + Executors = [], // Agents have empty executors list (workflows use this field) + }; + } + + private static EntityInfo CreateWorkflowEntityInfo(Workflow workflow) + { + // Extract executor IDs from the workflow structure + var executorIds = new HashSet { workflow.StartExecutorId }; + var reflectedEdges = workflow.ReflectEdges(); + foreach (var (sourceId, edgeSet) in reflectedEdges) + { + executorIds.Add(sourceId); + foreach (var edge in edgeSet) + { + foreach (var sinkId in edge.Connection.SinkIds) + { + executorIds.Add(sinkId); + } + } + } + + // Create a default input schema (string type) + var defaultInputSchema = new Dictionary + { + ["type"] = "string" + }; + + var workflowId = workflow.Name ?? workflow.StartExecutorId; + return new EntityInfo( + Id: workflowId, + Type: "workflow", + Name: workflowId, + Description: workflow.Description, + Framework: "agent_framework", + Tools: [], + Metadata: [] + ) + { + Source = "in_memory", + Executors = [.. executorIds], // Workflows use Executors instead of Tools + WorkflowDump = JsonSerializer.SerializeToElement( + workflow.ToDevUIDict(), + EntitiesJsonContext.Default.DictionaryStringJsonElement), + InputSchema = JsonSerializer.SerializeToElement(defaultInputSchema, EntitiesJsonContext.Default.DictionaryStringString), + InputTypeName = "string", + StartExecutorId = workflow.StartExecutorId + }; + } + + private static IEnumerable GetRegisteredEntities(IServiceProvider serviceProvider) + { + var keyedEntities = serviceProvider.GetKeyedServices(KeyedService.AnyKey); + var defaultEntities = serviceProvider.GetServices() ?? []; + + return keyedEntities + .Concat(defaultEntities) + .Where(entity => entity is not null); + } } diff --git a/dotnet/src/Microsoft.Agents.AI.DevUI/HostApplicationBuilderExtensions.cs b/dotnet/src/Microsoft.Agents.AI.DevUI/HostApplicationBuilderExtensions.cs new file mode 100644 index 0000000000..30fa9ad29e --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DevUI/HostApplicationBuilderExtensions.cs @@ -0,0 +1,23 @@ +// Copyright (c) Microsoft. All rights reserved. + +namespace Microsoft.Extensions.Hosting; + +/// +/// Extension methods for to configure DevUI. +/// +public static class MicrosoftAgentAIDevUIHostApplicationBuilderExtensions +{ + /// + /// Adds DevUI services to the host application builder. + /// + /// The to configure. + /// The for method chaining. + public static IHostApplicationBuilder AddDevUI(this IHostApplicationBuilder builder) + { + ArgumentNullException.ThrowIfNull(builder); + + builder.Services.AddDevUI(); + + return builder; + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.DevUI/MetaApiExtensions.cs b/dotnet/src/Microsoft.Agents.AI.DevUI/MetaApiExtensions.cs new file mode 100644 index 0000000000..4a3cfbb8f0 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DevUI/MetaApiExtensions.cs @@ -0,0 +1,61 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Microsoft.Agents.AI.DevUI.Entities; + +namespace Microsoft.Agents.AI.DevUI; + +/// +/// Provides extension methods for mapping the server metadata endpoint to an . +/// +internal static class MetaApiExtensions +{ + /// + /// Maps the HTTP API endpoint for retrieving server metadata. + /// + /// The to add the route to. + /// The for method chaining. + /// + /// This extension method registers the following endpoint: + /// + /// GET /meta - Retrieve server metadata including UI mode, version, capabilities, and auth requirements + /// + /// The endpoint is compatible with the Python DevUI frontend and provides essential + /// configuration information needed for proper frontend initialization. + /// + public static IEndpointConventionBuilder MapMeta(this IEndpointRouteBuilder endpoints) + { + return endpoints.MapGet("/meta", GetMeta) + .WithName("GetMeta") + .WithSummary("Get server metadata and configuration") + .WithDescription("Returns server metadata including UI mode, version, framework identifier, capabilities, and authentication requirements. Used by the frontend for initialization and feature detection.") + .Produces(StatusCodes.Status200OK, contentType: "application/json"); + } + + private static IResult GetMeta() + { + // TODO: Consider making these configurable via IOptions + // For now, using sensible defaults that match Python DevUI behavior + + var meta = new MetaResponse + { + UiMode = "developer", // Could be made configurable to support "user" mode + Version = "0.1.0", // TODO: Extract from assembly version attribute + Framework = "agent_framework", + Runtime = "dotnet", // .NET runtime for deployment guides + Capabilities = new Dictionary + { + // Tracing capability - will be enabled when trace event support is added + ["tracing"] = false, + + // OpenAI proxy capability - not currently supported in .NET DevUI + ["openai_proxy"] = false, + + // Deployment capability - not currently supported in .NET DevUI + ["deployment"] = false + }, + AuthRequired = false // Could be made configurable based on authentication middleware + }; + + return Results.Json(meta, EntitiesJsonContext.Default.MetaResponse); + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.DevUI/Microsoft.Agents.AI.DevUI.Frontend.targets b/dotnet/src/Microsoft.Agents.AI.DevUI/Microsoft.Agents.AI.DevUI.Frontend.targets index f62a92e28d..c8bdc5dbdf 100644 --- a/dotnet/src/Microsoft.Agents.AI.DevUI/Microsoft.Agents.AI.DevUI.Frontend.targets +++ b/dotnet/src/Microsoft.Agents.AI.DevUI/Microsoft.Agents.AI.DevUI.Frontend.targets @@ -8,11 +8,6 @@ $(FrontendRoot)\node_modules - - - - - @@ -27,19 +22,6 @@ - - - $(BaseIntermediateOutputPath)\frontend.build.marker - - - - - - - - - - @@ -48,7 +30,7 @@ - + diff --git a/dotnet/src/Microsoft.Agents.AI.DevUI/Microsoft.Agents.AI.DevUI.csproj b/dotnet/src/Microsoft.Agents.AI.DevUI/Microsoft.Agents.AI.DevUI.csproj index 37aa6c37f8..30943cb5c4 100644 --- a/dotnet/src/Microsoft.Agents.AI.DevUI/Microsoft.Agents.AI.DevUI.csproj +++ b/dotnet/src/Microsoft.Agents.AI.DevUI/Microsoft.Agents.AI.DevUI.csproj @@ -1,7 +1,7 @@  - net9.0 + $(TargetFrameworksCore) enable enable Microsoft.Agents.AI.DevUI @@ -12,6 +12,10 @@ $(NoWarn);CS1591;CA1852;CA1050;RCS1037;RCS1036;RCS1124;RCS1021;RCS1146;RCS1211;CA2007;CA1308;IL2026;IL3050;CA1812 + + true + + @@ -23,14 +27,13 @@ - - - - Microsoft Agent Framework Developer UI Provides Microsoft Agent Framework support for developer UI. + + + diff --git a/dotnet/src/Microsoft.Agents.AI.DevUI/README.md b/dotnet/src/Microsoft.Agents.AI.DevUI/README.md index 1f106e29ef..104c43729b 100644 --- a/dotnet/src/Microsoft.Agents.AI.DevUI/README.md +++ b/dotnet/src/Microsoft.Agents.AI.DevUI/README.md @@ -24,14 +24,22 @@ var builder = WebApplication.CreateBuilder(args); // Register your agents builder.AddAIAgent("assistant", "You are a helpful assistant."); +// Register DevUI services if (builder.Environment.IsDevelopment()) { - // Add DevUI services builder.AddDevUI(); } +// Register services for OpenAI responses and conversations (also required for DevUI) +builder.AddOpenAIResponses(); +builder.AddOpenAIConversations(); + var app = builder.Build(); +// Map endpoints for OpenAI responses and conversations (also required for DevUI) +app.MapOpenAIResponses(); +app.MapOpenAIConversations(); + if (builder.Environment.IsDevelopment()) { // Map DevUI endpoint to /devui diff --git a/dotnet/src/Microsoft.Agents.AI.DevUI/ServiceCollectionsExtensions.cs b/dotnet/src/Microsoft.Agents.AI.DevUI/ServiceCollectionsExtensions.cs new file mode 100644 index 0000000000..6971e3d2e0 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DevUI/ServiceCollectionsExtensions.cs @@ -0,0 +1,61 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Microsoft.Agents.AI; +using Microsoft.Agents.AI.Workflows; +using Microsoft.Shared.Diagnostics; + +namespace Microsoft.Extensions.DependencyInjection; + +/// +/// Extension methods for to configure DevUI. +/// +public static class MicrosoftAgentAIDevUIServiceCollectionsExtensions +{ + /// + /// Adds services required for DevUI integration. + /// + /// The to configure. + /// The for method chaining. + public static IServiceCollection AddDevUI(this IServiceCollection services) + { + ArgumentNullException.ThrowIfNull(services); + + // a factory that tries to construct an AIAgent from Workflow, + // even if workflow was not explicitly registered as an AIAgent. + +#pragma warning disable IDE0001 // Simplify Names + services.AddKeyedSingleton(KeyedService.AnyKey, (sp, key) => + { + var keyAsStr = key as string; + Throw.IfNullOrEmpty(keyAsStr); + + var workflow = sp.GetKeyedService(keyAsStr); + if (workflow is not null) + { + return workflow.AsAgent(name: workflow.Name); + } + + // another thing we can do is resolve a non-keyed workflow. + // however, we can't rely on anything than key to be equal to the workflow.Name. + // so we try: if we fail, we return null. + workflow = sp.GetService(); + if (workflow is not null && workflow.Name?.Equals(keyAsStr, StringComparison.Ordinal) == true) + { + return workflow.AsAgent(name: workflow.Name); + } + + // and it's possible to lookup at the default-registered AIAgent + // with the condition of same name as the key. + var agent = sp.GetService(); + if (agent is not null && agent.Name?.Equals(keyAsStr, StringComparison.Ordinal) == true) + { + return agent; + } + + return null!; + }); +#pragma warning restore IDE0001 // Simplify Names + + return services; + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.DurableTask/AIAgentExtensions.cs b/dotnet/src/Microsoft.Agents.AI.DurableTask/AIAgentExtensions.cs new file mode 100644 index 0000000000..5eac1b84e0 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DurableTask/AIAgentExtensions.cs @@ -0,0 +1,47 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Microsoft.Extensions.AI; +using Microsoft.Extensions.DependencyInjection; + +namespace Microsoft.Agents.AI.DurableTask; + +/// +/// Extension methods for the class. +/// +public static class AIAgentExtensions +{ + /// + /// Converts an AIAgent to a durable agent proxy. + /// + /// The agent to convert. + /// The service provider. + /// The durable agent proxy. + /// + /// Thrown when the agent is a instance or if the agent has no name. + /// + /// + /// Thrown if does not contain an + /// or if durable agents have not been configured on the service collection. + /// + /// + /// Thrown when the agent with the specified name has not been registered. + /// + public static AIAgent AsDurableAgentProxy(this AIAgent agent, IServiceProvider services) + { + // Don't allow this method to be used on DurableAIAgent instances. + if (agent is DurableAIAgent) + { + throw new ArgumentException( + $"{nameof(DurableAIAgent)} instances cannot be converted to a durable agent proxy.", + nameof(agent)); + } + + string agentName = agent.Name ?? throw new ArgumentException("Agent must have a name.", nameof(agent)); + + // Validate that the agent is registered + ServiceCollectionExtensions.ValidateAgentIsRegistered(services, agentName); + + IDurableAgentClient agentClient = services.GetRequiredService(); + return new DurableAIAgentProxy(agentName, agentClient); + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.DurableTask/AgentEntity.cs b/dotnet/src/Microsoft.Agents.AI.DurableTask/AgentEntity.cs new file mode 100644 index 0000000000..166799a124 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DurableTask/AgentEntity.cs @@ -0,0 +1,124 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Microsoft.Agents.AI.DurableTask.State; +using Microsoft.DurableTask.Client; +using Microsoft.DurableTask.Entities; +using Microsoft.Extensions.AI; +using Microsoft.Extensions.DependencyInjection; +using Microsoft.Extensions.Hosting; +using Microsoft.Extensions.Logging; + +namespace Microsoft.Agents.AI.DurableTask; + +internal class AgentEntity(IServiceProvider services, CancellationToken cancellationToken = default) : TaskEntity +{ + private readonly IServiceProvider _services = services; + private readonly DurableTaskClient _client = services.GetRequiredService(); + private readonly ILoggerFactory _loggerFactory = services.GetRequiredService(); + private readonly IAgentResponseHandler? _messageHandler = services.GetService(); + private readonly CancellationToken _cancellationToken = cancellationToken != default + ? cancellationToken + : services.GetService()?.ApplicationStopping ?? CancellationToken.None; + + public async Task RunAgentAsync(RunRequest request) + { + AgentSessionId sessionId = this.Context.Id; + IReadOnlyDictionary> agents = + this._services.GetRequiredService>>(); + if (!agents.TryGetValue(sessionId.Name, out Func? agentFactory)) + { + throw new InvalidOperationException($"Agent '{sessionId.Name}' not found"); + } + + AIAgent agent = agentFactory(this._services); + EntityAgentWrapper agentWrapper = new(agent, this.Context, request, this._services); + + // Logger category is Microsoft.DurableTask.Agents.{agentName}.{sessionId} + ILogger logger = this._loggerFactory.CreateLogger($"Microsoft.DurableTask.Agents.{agent.Name}.{sessionId.Key}"); + + if (request.Messages.Count == 0) + { + logger.LogInformation("Ignoring empty request"); + } + + this.State.Data.ConversationHistory.Add(DurableAgentStateRequest.FromRunRequest(request)); + + foreach (ChatMessage msg in request.Messages) + { + logger.LogAgentRequest(sessionId, msg.Role, msg.Text); + } + + // Set the current agent context for the duration of the agent run. This will be exposed + // to any tools that are invoked by the agent. + DurableAgentContext agentContext = new( + entityContext: this.Context, + client: this._client, + lifetime: this._services.GetRequiredService(), + services: this._services); + DurableAgentContext.SetCurrent(agentContext); + + try + { + // Start the agent response stream + IAsyncEnumerable responseStream = agentWrapper.RunStreamingAsync( + this.State.Data.ConversationHistory.SelectMany(e => e.Messages).Select(m => m.ToChatMessage()), + agentWrapper.GetNewThread(), + options: null, + this._cancellationToken); + + AgentRunResponse response; + if (this._messageHandler is null) + { + // If no message handler is provided, we can just get the full response at once. + // This is expected to be the common case for non-interactive agents. + response = await responseStream.ToAgentRunResponseAsync(this._cancellationToken); + } + else + { + List responseUpdates = []; + + // To support interactive chat agents, we need to stream the responses to an IAgentMessageHandler. + // The user-provided message handler can be implemented to send the responses to the user. + // We assume that only non-empty text updates are useful for the user. + async IAsyncEnumerable StreamResultsAsync() + { + await foreach (AgentRunResponseUpdate update in responseStream) + { + // We need the full response further down, so we piece it together as we go. + responseUpdates.Add(update); + + // Yield the update to the message handler. + yield return update; + } + } + + await this._messageHandler.OnStreamingResponseUpdateAsync(StreamResultsAsync(), this._cancellationToken); + response = responseUpdates.ToAgentRunResponse(); + } + + // Persist the agent response to the entity state for client polling + this.State.Data.ConversationHistory.Add( + DurableAgentStateResponse.FromRunResponse(request.CorrelationId, response)); + + string responseText = response.Text; + + if (!string.IsNullOrEmpty(responseText)) + { + logger.LogAgentResponse( + sessionId, + response.Messages.FirstOrDefault()?.Role ?? ChatRole.Assistant, + responseText, + response.Usage?.InputTokenCount, + response.Usage?.OutputTokenCount, + response.Usage?.TotalTokenCount); + } + + return response; + } + finally + { + // Clear the current agent context + DurableAgentContext.ClearCurrent(); + } + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.DurableTask/AgentNotRegisteredException.cs b/dotnet/src/Microsoft.Agents.AI.DurableTask/AgentNotRegisteredException.cs new file mode 100644 index 0000000000..fc051fa0b2 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DurableTask/AgentNotRegisteredException.cs @@ -0,0 +1,47 @@ +// Copyright (c) Microsoft. All rights reserved. + +namespace Microsoft.Agents.AI.DurableTask; + +/// +/// Exception thrown when an agent with the specified name has not been registered. +/// +public sealed class AgentNotRegisteredException : InvalidOperationException +{ + // Not used, but required by static analysis. + private AgentNotRegisteredException() + { + this.AgentName = string.Empty; + } + + /// + /// Initializes a new instance of the class with the agent name. + /// + /// The name of the agent that was not registered. + public AgentNotRegisteredException(string agentName) + : base(GetMessage(agentName)) + { + this.AgentName = agentName; + } + + /// + /// Initializes a new instance of the class with the agent name and an inner exception. + /// + /// The name of the agent that was not registered. + /// The exception that is the cause of the current exception. + public AgentNotRegisteredException(string agentName, Exception? innerException) + : base(GetMessage(agentName), innerException) + { + this.AgentName = agentName; + } + + /// + /// Gets the name of the agent that was not registered. + /// + public string AgentName { get; } + + private static string GetMessage(string agentName) + { + ArgumentException.ThrowIfNullOrEmpty(agentName); + return $"No agent named '{agentName}' was registered. Ensure the agent is registered using {nameof(ServiceCollectionExtensions.ConfigureDurableAgents)} before using it in an orchestration."; + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.DurableTask/AgentRunHandle.cs b/dotnet/src/Microsoft.Agents.AI.DurableTask/AgentRunHandle.cs new file mode 100644 index 0000000000..e4fe08dbf2 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DurableTask/AgentRunHandle.cs @@ -0,0 +1,83 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Microsoft.Agents.AI.DurableTask.State; +using Microsoft.DurableTask.Client; +using Microsoft.DurableTask.Client.Entities; +using Microsoft.Extensions.Logging; + +namespace Microsoft.Agents.AI.DurableTask; + +/// +/// Represents a handle for a running agent request that can be used to retrieve the response. +/// +internal sealed class AgentRunHandle +{ + private readonly DurableTaskClient _client; + private readonly ILogger _logger; + + internal AgentRunHandle( + DurableTaskClient client, + ILogger logger, + AgentSessionId sessionId, + string correlationId) + { + this._client = client; + this._logger = logger; + this.SessionId = sessionId; + this.CorrelationId = correlationId; + } + + /// + /// Gets the correlation ID for this request. + /// + public string CorrelationId { get; } + + /// + /// Gets the session ID for this request. + /// + public AgentSessionId SessionId { get; } + + /// + /// Reads the agent response for this request by polling the entity state until the response is found. + /// Uses an exponential backoff polling strategy with a maximum interval of 1 second. + /// + /// The cancellation token. + /// The agent response corresponding to this request. + /// Thrown when the response is not found after polling. + public async Task ReadAgentResponseAsync(CancellationToken cancellationToken = default) + { + TimeSpan pollInterval = TimeSpan.FromMilliseconds(50); // Start with 50ms + TimeSpan maxPollInterval = TimeSpan.FromSeconds(3); // Maximum 3 seconds + + this._logger.LogStartPollingForResponse(this.SessionId, this.CorrelationId); + + while (true) + { + // Poll the entity state for responses + EntityMetadata? entityResponse = await this._client.Entities.GetEntityAsync( + this.SessionId, + cancellation: cancellationToken); + DurableAgentState? state = entityResponse?.State; + + if (state?.Data.ConversationHistory is not null) + { + // Look for an agent response with matching CorrelationId + DurableAgentStateResponse? response = state.Data.ConversationHistory + .OfType() + .FirstOrDefault(r => r.CorrelationId == this.CorrelationId); + + if (response is not null) + { + this._logger.LogDonePollingForResponse(this.SessionId, this.CorrelationId); + return response.ToRunResponse(); + } + } + + // Wait before polling again with exponential backoff + await Task.Delay(pollInterval, cancellationToken); + + // Double the poll interval, but cap it at the maximum + pollInterval = TimeSpan.FromMilliseconds(Math.Min(pollInterval.TotalMilliseconds * 2, maxPollInterval.TotalMilliseconds)); + } + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.DurableTask/AgentSessionId.cs b/dotnet/src/Microsoft.Agents.AI.DurableTask/AgentSessionId.cs new file mode 100644 index 0000000000..f183ec84dc --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DurableTask/AgentSessionId.cs @@ -0,0 +1,165 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json; +using System.Text.Json.Serialization; +using Microsoft.DurableTask.Entities; + +namespace Microsoft.Agents.AI.DurableTask; + +/// +/// Represents an agent session ID, which is used to identify a long-running agent session. +/// +[JsonConverter(typeof(AgentSessionIdJsonConverter))] +public readonly struct AgentSessionId : IEquatable +{ + private const string EntityNamePrefix = "dafx-"; + private readonly EntityInstanceId _entityId; + + /// + /// Initializes a new instance of the struct. + /// + /// The name of the agent that owns the session (case-insensitive). + /// The unique key of the agent session (case-sensitive). + public AgentSessionId(string name, string key) + { + this.Name = name; + this._entityId = new EntityInstanceId(ToEntityName(name), key); + } + + /// + /// Converts an agent name to its underlying entity name representation. + /// + /// The agent name. + /// The entity name used by Durable Task for this agent. + public static string ToEntityName(string name) => $"{EntityNamePrefix}{name}"; + + /// + /// Gets the name of the agent that owns the session. Names are case-insensitive. + /// + public string Name { get; } + + /// + /// Gets the unique key of the agent session. Keys are case-sensitive and are used to identify the session. + /// + public string Key => this._entityId.Key; + + internal EntityInstanceId ToEntityId() => this._entityId; + + /// + /// Creates a new with the specified name and a randomly generated key. + /// + /// The name of the agent that owns the session. + /// A new with the specified name and a random key. + public static AgentSessionId WithRandomKey(string name) => + new(name, Guid.NewGuid().ToString("N")); + + /// + /// Determines whether two instances are equal. + /// + /// The first to compare. + /// The second to compare. + /// true if the two instances are equal; otherwise, false. + public static bool operator ==(AgentSessionId left, AgentSessionId right) => + left._entityId == right._entityId; + + /// + /// Determines whether two instances are not equal. + /// + /// The first to compare. + /// The second to compare. + /// true if the two instances are not equal; otherwise, false. + public static bool operator !=(AgentSessionId left, AgentSessionId right) => + left._entityId != right._entityId; + + /// + /// Determines whether the specified is equal to the current . + /// + /// The to compare with the current . + /// true if the specified is equal to the current ; otherwise, false. + public bool Equals(AgentSessionId other) => this == other; + + /// + /// Determines whether the specified object is equal to the current . + /// + /// The object to compare with the current . + /// true if the specified object is equal to the current ; otherwise, false. + public override bool Equals(object? obj) => obj is AgentSessionId other && this == other; + + /// + /// Returns the hash code for this . + /// + /// A hash code for the current . + public override int GetHashCode() => this._entityId.GetHashCode(); + + /// + /// Returns a string representation of this in the form of @name@key. + /// + /// A string representation of the current . + public override string ToString() => this._entityId.ToString(); + + /// + /// Converts the string representation of an agent session ID to its equivalent. + /// The input string must be in the form of @name@key. + /// + /// A string containing an agent session ID to convert. + /// A equivalent to the agent session ID contained in . + /// Thrown when is not a valid agent session ID format. + public static AgentSessionId Parse(string sessionIdString) + { + EntityInstanceId entityId = EntityInstanceId.FromString(sessionIdString); + if (!entityId.Name.StartsWith(EntityNamePrefix, StringComparison.OrdinalIgnoreCase)) + { + throw new ArgumentException($"'{sessionIdString}' is not a valid agent session ID.", nameof(sessionIdString)); + } + + return new AgentSessionId(entityId.Name[EntityNamePrefix.Length..], entityId.Key); + } + + /// + /// Implicitly converts an to an . + /// This conversion is useful for entity API interoperability. + /// + /// The to convert. + /// The equivalent . + public static implicit operator EntityInstanceId(AgentSessionId agentSessionId) => agentSessionId.ToEntityId(); + + /// + /// Implicitly converts an to an . + /// + /// The to convert. + /// The equivalent . + [System.Diagnostics.CodeAnalysis.SuppressMessage("Design", "CA1065:Do not raise exceptions in unexpected locations", Justification = "Implicit conversion must validate format.")] + public static implicit operator AgentSessionId(EntityInstanceId entityId) + { + if (!entityId.Name.StartsWith(EntityNamePrefix, StringComparison.OrdinalIgnoreCase)) + { + throw new ArgumentException($"'{entityId}' is not a valid agent session ID.", nameof(entityId)); + } + return new AgentSessionId(entityId.Name[EntityNamePrefix.Length..], entityId.Key); + } + + /// + /// Custom JSON converter for to ensure proper serialization and deserialization. + /// + public sealed class AgentSessionIdJsonConverter : JsonConverter + { + /// + public override AgentSessionId Read(ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options) + { + if (reader.TokenType != JsonTokenType.String) + { + throw new JsonException("Expected string value"); + } + + string value = reader.GetString() ?? string.Empty; + + return Parse(value); + } + + /// + public override void Write(Utf8JsonWriter writer, AgentSessionId value, JsonSerializerOptions options) + { + writer.WriteStringValue(value.ToString()); + } + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.DurableTask/CHANGELOG.md b/dotnet/src/Microsoft.Agents.AI.DurableTask/CHANGELOG.md new file mode 100644 index 0000000000..d2cdc7cd41 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DurableTask/CHANGELOG.md @@ -0,0 +1,17 @@ +# Release History + +## v1.0.0-preview.251204.1 + +- Added orchestration ID to durable agent entity state ([#2137](https://github.com/microsoft/agent-framework/pull/2137)) + +## v1.0.0-preview.251125.1 + +- Added support for .NET 10 ([#2128](https://github.com/microsoft/agent-framework/pull/2128)) + +## v1.0.0-preview.251114.1 + +- Added friendly error message when running durable agent that isn't registered ([#2214](https://github.com/microsoft/agent-framework/pull/2214)) + +## v1.0.0-preview.251112.1 + +- Initial public release ([#1916](https://github.com/microsoft/agent-framework/pull/1916)) diff --git a/dotnet/src/Microsoft.Agents.AI.DurableTask/DefaultDurableAgentClient.cs b/dotnet/src/Microsoft.Agents.AI.DurableTask/DefaultDurableAgentClient.cs new file mode 100644 index 0000000000..2086a00ecb --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DurableTask/DefaultDurableAgentClient.cs @@ -0,0 +1,31 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Microsoft.DurableTask.Client; +using Microsoft.Extensions.Logging; +using Microsoft.Extensions.Logging.Abstractions; + +namespace Microsoft.Agents.AI.DurableTask; + +internal class DefaultDurableAgentClient(DurableTaskClient client, ILoggerFactory loggerFactory) : IDurableAgentClient +{ + private readonly DurableTaskClient _client = client ?? throw new ArgumentNullException(nameof(client)); + private readonly ILogger _logger = (loggerFactory ?? NullLoggerFactory.Instance).CreateLogger(); + + public async Task RunAgentAsync( + AgentSessionId sessionId, + RunRequest request, + CancellationToken cancellationToken = default) + { + ArgumentNullException.ThrowIfNull(request); + + this._logger.LogSignallingAgent(sessionId); + + await this._client.Entities.SignalEntityAsync( + sessionId, + nameof(AgentEntity.RunAgentAsync), + request, + cancellation: cancellationToken); + + return new AgentRunHandle(this._client, this._logger, sessionId, request.CorrelationId); + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.DurableTask/DurableAIAgent.cs b/dotnet/src/Microsoft.Agents.AI.DurableTask/DurableAIAgent.cs new file mode 100644 index 0000000000..021c8f22c7 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DurableTask/DurableAIAgent.cs @@ -0,0 +1,251 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Diagnostics.CodeAnalysis; +using System.Runtime.CompilerServices; +using System.Text.Json; +using System.Text.Json.Serialization.Metadata; +using Microsoft.DurableTask; +using Microsoft.DurableTask.Entities; +using Microsoft.Extensions.AI; + +namespace Microsoft.Agents.AI.DurableTask; + +/// +/// A durable AIAgent implementation that uses entity methods to interact with agent entities. +/// +public sealed class DurableAIAgent : AIAgent +{ + private readonly TaskOrchestrationContext _context; + private readonly string _agentName; + + /// + /// Initializes a new instance of the class. + /// + /// The orchestration context. + /// The name of the agent. + internal DurableAIAgent(TaskOrchestrationContext context, string agentName) + { + this._context = context; + this._agentName = agentName; + } + + /// + /// Creates a new agent thread for this agent using a random session ID. + /// + /// A new agent thread. + public override AgentThread GetNewThread() + { + AgentSessionId sessionId = this._context.NewAgentSessionId(this._agentName); + return new DurableAgentThread(sessionId); + } + + /// + /// Deserializes an agent thread from JSON. + /// + /// The serialized thread data. + /// Optional JSON serializer options. + /// The deserialized agent thread. + public override AgentThread DeserializeThread( + JsonElement serializedThread, + JsonSerializerOptions? jsonSerializerOptions = null) + { + return DurableAgentThread.Deserialize(serializedThread, jsonSerializerOptions); + } + + /// + /// Runs the agent with messages and returns the response. + /// + /// The messages to send to the agent. + /// The agent thread to use. + /// Optional run options. + /// The cancellation token. + /// The response from the agent. + /// Thrown when the agent has not been registered. + /// Thrown when the provided thread is not valid for a durable agent. + /// Thrown when cancellation is requested (cancellation is not supported for durable agents). + public override async Task RunAsync( + IEnumerable messages, + AgentThread? thread = null, + AgentRunOptions? options = null, + CancellationToken cancellationToken = default) + { + if (cancellationToken != default && cancellationToken.CanBeCanceled) + { + throw new NotSupportedException("Cancellation is not supported for durable agents."); + } + + thread ??= this.GetNewThread(); + if (thread is not DurableAgentThread durableThread) + { + throw new ArgumentException( + "The provided thread is not valid for a durable agent. " + + "Create a new thread using GetNewThread or provide a thread previously created by this agent.", + paramName: nameof(thread)); + } + + IList? enableToolNames = null; + bool enableToolCalls = true; + ChatResponseFormat? responseFormat = null; + if (options is DurableAgentRunOptions durableOptions) + { + enableToolCalls = durableOptions.EnableToolCalls; + enableToolNames = durableOptions.EnableToolNames; + responseFormat = durableOptions.ResponseFormat; + } + else if (options is ChatClientAgentRunOptions chatClientOptions && chatClientOptions.ChatOptions?.Tools != null) + { + // Honor the response format from the chat client options if specified + responseFormat = chatClientOptions.ChatOptions?.ResponseFormat; + } + + RunRequest request = new([.. messages], responseFormat, enableToolCalls, enableToolNames) + { + OrchestrationId = this._context.InstanceId + }; + + try + { + return await this._context.Entities.CallEntityAsync( + durableThread.SessionId, + nameof(AgentEntity.RunAgentAsync), + request); + } + catch (EntityOperationFailedException e) when (e.FailureDetails.ErrorType == "EntityTaskNotFound") + { + throw new AgentNotRegisteredException(this._agentName, e); + } + } + + /// + /// Runs the agent with messages and returns a simulated streaming response. + /// + /// + /// Streaming is not supported for durable agents, so this method just returns the full response + /// as a single update. + /// + /// The messages to send to the agent. + /// The agent thread to use. + /// Optional run options. + /// The cancellation token. + /// A streaming response enumerable. + public override async IAsyncEnumerable RunStreamingAsync( + IEnumerable messages, + AgentThread? thread = null, + AgentRunOptions? options = null, + [EnumeratorCancellation] CancellationToken cancellationToken = default) + { + // Streaming is not supported for durable agents, so we just return the full response + // as a single update. + AgentRunResponse response = await this.RunAsync(messages, thread, options, cancellationToken); + foreach (AgentRunResponseUpdate update in response.ToAgentRunResponseUpdates()) + { + yield return update; + } + } + + /// + /// Runs the agent with a message and returns the deserialized output as an instance of . + /// + /// The message to send to the agent. + /// The agent thread to use. + /// Optional JSON serializer options. + /// Optional run options. + /// The cancellation token. + /// The type of the output. + /// + /// Thrown when the provided already contains a response schema. + /// Thrown when the provided is not a . + /// + /// + /// Thrown when the agent response is empty or cannot be deserialized. + /// + /// The output from the agent. + public async Task> RunAsync( + string message, + AgentThread? thread = null, + JsonSerializerOptions? serializerOptions = null, + AgentRunOptions? options = null, + CancellationToken cancellationToken = default) + { + return await this.RunAsync( + messages: [new ChatMessage(ChatRole.User, message) { CreatedAt = DateTimeOffset.UtcNow }], + thread, + serializerOptions, + options, + cancellationToken); + } + + /// + /// Runs the agent with messages and returns the deserialized output as an instance of . + /// + /// The messages to send to the agent. + /// The agent thread to use. + /// Optional JSON serializer options. + /// Optional run options. + /// The cancellation token. + /// The type of the output. + /// + /// Thrown when the provided already contains a response schema. + /// Thrown when the provided is not a . + /// + /// + /// Thrown when the agent response is empty or cannot be deserialized. + /// + /// The output from the agent. + [UnconditionalSuppressMessage("Trimming", "IL2026", Justification = "Fallback to reflection-based deserialization is intentional for library flexibility with user-defined types.")] + [UnconditionalSuppressMessage("ReflectionAnalysis", "IL3050", Justification = "Fallback to reflection-based deserialization is intentional for library flexibility with user-defined types.")] + public async Task> RunAsync( + IEnumerable messages, + AgentThread? thread = null, + JsonSerializerOptions? serializerOptions = null, + AgentRunOptions? options = null, + CancellationToken cancellationToken = default) + { + options ??= new DurableAgentRunOptions(); + if (options is not DurableAgentRunOptions durableOptions) + { + throw new ArgumentException( + "Response schema is only supported with DurableAgentRunOptions when using durable agents. " + + "Cannot specify a response schema when calling RunAsync.", + paramName: nameof(options)); + } + + if (durableOptions.ResponseFormat is not null) + { + throw new ArgumentException( + "A response schema is already defined in the provided DurableAgentRunOptions. " + + "Cannot specify a response schema when calling RunAsync.", + paramName: nameof(options)); + } + + // Create the JSON schema for the response type + durableOptions.ResponseFormat = ChatResponseFormat.ForJsonSchema(); + + AgentRunResponse response = await this.RunAsync(messages, thread, durableOptions, cancellationToken); + + // Deserialize the response text to the requested type + if (string.IsNullOrEmpty(response.Text)) + { + throw new InvalidOperationException("Agent response is empty and cannot be deserialized."); + } + + serializerOptions ??= DurableAgentJsonUtilities.DefaultOptions; + + // Prefer source-generated metadata when available to support AOT/trimming scenarios. + // Fallback to reflection-based deserialization for types without source-generated metadata. + // This is necessary since T is a user-provided type that may not have [JsonSerializable] coverage. + JsonTypeInfo? typeInfo = serializerOptions.GetTypeInfo(typeof(T)); + T? result = (typeInfo is JsonTypeInfo typedInfo + ? (T?)JsonSerializer.Deserialize(response.Text, typedInfo) + : JsonSerializer.Deserialize(response.Text, serializerOptions)) + ?? throw new InvalidOperationException($"Failed to deserialize agent response to type {typeof(T).Name}."); + + return new DurableAIAgentRunResponse(response, result); + } + + private sealed class DurableAIAgentRunResponse(AgentRunResponse response, T result) + : AgentRunResponse(response.AsChatResponse()) + { + public override T Result { get; } = result; + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.DurableTask/DurableAIAgentProxy.cs b/dotnet/src/Microsoft.Agents.AI.DurableTask/DurableAIAgentProxy.cs new file mode 100644 index 0000000000..58f9598a7e --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DurableTask/DurableAIAgentProxy.cs @@ -0,0 +1,81 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json; +using Microsoft.Extensions.AI; + +namespace Microsoft.Agents.AI.DurableTask; + +internal class DurableAIAgentProxy(string name, IDurableAgentClient agentClient) : AIAgent +{ + private readonly IDurableAgentClient _agentClient = agentClient; + + public override string? Name { get; } = name; + + public override AgentThread DeserializeThread( + JsonElement serializedThread, + JsonSerializerOptions? jsonSerializerOptions = null) + { + return DurableAgentThread.Deserialize(serializedThread, jsonSerializerOptions); + } + + public override AgentThread GetNewThread() + { + return new DurableAgentThread(AgentSessionId.WithRandomKey(this.Name!)); + } + + public override async Task RunAsync( + IEnumerable messages, + AgentThread? thread = null, + AgentRunOptions? options = null, + CancellationToken cancellationToken = default) + { + thread ??= this.GetNewThread(); + if (thread is not DurableAgentThread durableThread) + { + throw new ArgumentException( + "The provided thread is not valid for a durable agent. " + + "Create a new thread using GetNewThread or provide a thread previously created by this agent.", + paramName: nameof(thread)); + } + + IList? enableToolNames = null; + bool enableToolCalls = true; + ChatResponseFormat? responseFormat = null; + bool isFireAndForget = false; + + if (options is DurableAgentRunOptions durableOptions) + { + enableToolCalls = durableOptions.EnableToolCalls; + enableToolNames = durableOptions.EnableToolNames; + responseFormat = durableOptions.ResponseFormat; + isFireAndForget = durableOptions.IsFireAndForget; + } + else if (options is ChatClientAgentRunOptions chatClientOptions) + { + // Honor the response format from the chat client options if specified + responseFormat = chatClientOptions.ChatOptions?.ResponseFormat; + } + + RunRequest request = new([.. messages], responseFormat, enableToolCalls, enableToolNames); + AgentSessionId sessionId = durableThread.SessionId; + + AgentRunHandle agentRunHandle = await this._agentClient.RunAgentAsync(sessionId, request, cancellationToken); + + if (isFireAndForget) + { + // If the request is fire and forget, return an empty response. + return new AgentRunResponse(); + } + + return await agentRunHandle.ReadAgentResponseAsync(cancellationToken); + } + + public override IAsyncEnumerable RunStreamingAsync( + IEnumerable messages, + AgentThread? thread = null, + AgentRunOptions? options = null, + CancellationToken cancellationToken = default) + { + throw new NotSupportedException("Streaming is not supported for durable agents."); + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.DurableTask/DurableAgentContext.cs b/dotnet/src/Microsoft.Agents.AI.DurableTask/DurableAgentContext.cs new file mode 100644 index 0000000000..94a6c00424 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DurableTask/DurableAgentContext.cs @@ -0,0 +1,161 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Microsoft.DurableTask; +using Microsoft.DurableTask.Client; +using Microsoft.DurableTask.Entities; +using Microsoft.Extensions.DependencyInjection; +using Microsoft.Extensions.Hosting; + +namespace Microsoft.Agents.AI.DurableTask; + +/// +/// A context for durable agents that provides access to orchestration capabilities. +/// This class provides thread-static access to the current agent context. +/// +public class DurableAgentContext +{ + private static readonly AsyncLocal s_currentContext = new(); + private readonly IServiceProvider _services; + private readonly CancellationToken _cancellationToken; + + internal DurableAgentContext( + TaskEntityContext entityContext, + DurableTaskClient client, + IHostApplicationLifetime lifetime, + IServiceProvider services) + { + this.EntityContext = entityContext; + this.CurrentThread = new DurableAgentThread(entityContext.Id); + this.Client = client; + this._services = services; + this._cancellationToken = lifetime.ApplicationStopping; + } + + /// + /// Gets the current durable agent context instance. + /// + /// Thrown when no agent context is available. + public static DurableAgentContext Current => s_currentContext.Value ?? + throw new InvalidOperationException("No agent context found!"); + + /// + /// Gets the entity context for this agent. + /// + public TaskEntityContext EntityContext { get; } + + /// + /// Gets the durable task client for this agent. + /// + public DurableTaskClient Client { get; } + + /// + /// Gets the current agent thread. + /// + public DurableAgentThread CurrentThread { get; } + + /// + /// Sets the current durable agent context instance. + /// This is called internally by the agent entity during execution. + /// + /// The context instance to set. + internal static void SetCurrent(DurableAgentContext context) + { + if (s_currentContext.Value is not null) + { + throw new InvalidOperationException("A DurableAgentContext has already been set for this AsyncLocal context."); + } + + s_currentContext.Value = context; + } + + /// + /// Clears the current durable agent context instance. + /// This is called internally by the agent entity after execution. + /// + internal static void ClearCurrent() + { + s_currentContext.Value = null; + } + + /// + /// Schedules a new orchestration instance. + /// + /// + /// When run in the context of a durable agent tool, the actual scheduling of the orchestration + /// occurs after the completion of the tool call. This allows the durable scheduling of the orchestration + /// and the agent state update to be committed atomically in a single transaction. + /// + /// The name of the orchestration to schedule. + /// The input to the orchestration. + /// The options for the orchestration. + /// The instance ID of the scheduled orchestration. + public string ScheduleNewOrchestration( + TaskName name, + object? input = null, + StartOrchestrationOptions? options = null) + { + return this.EntityContext.ScheduleNewOrchestration(name, input, options); + } + + /// + /// Gets the status of an orchestration instance. + /// + /// The instance ID of the orchestration to get the status of. + /// Whether to include detailed information about the orchestration. + /// The status of the orchestration. + public Task GetOrchestrationStatusAsync(string instanceId, bool includeDetails = false) + { + return this.Client.GetInstanceAsync(instanceId, includeDetails, this._cancellationToken); + } + + /// + /// Raises an event on an orchestration instance. + /// + /// The instance ID of the orchestration to raise the event on. + /// The name of the event to raise. + /// The data to send with the event. +#pragma warning disable CA1030 // Use events where appropriate + public Task RaiseOrchestrationEventAsync(string instanceId, string eventName, object? eventData = null) +#pragma warning restore CA1030 // Use events where appropriate + { + return this.Client.RaiseEventAsync(instanceId, eventName, eventData, this._cancellationToken); + } + + /// + /// Asks the for an object of the specified type, . + /// + /// The type of the object being requested. + /// An optional key to identify the service instance. + /// The service instance, or if the service is not found. + /// + /// Thrown when is not and the service provider does not support keyed services. + /// + public TService? GetService(object? serviceKey = null) + { + return this.GetService(typeof(TService), serviceKey) is TService service ? service : default; + } + + /// + /// Asks the for an object of the specified type, . + /// + /// The type of the object being requested. + /// An optional key to identify the service instance. + /// The service instance, or if the service is not found. + /// + /// Thrown when is not and the service provider does not support keyed services. + /// + public object? GetService(Type serviceType, object? serviceKey = null) + { + if (serviceKey is not null) + { + if (this._services is not IKeyedServiceProvider keyedServiceProvider) + { + throw new InvalidOperationException("The service provider does not support keyed services."); + } + + return keyedServiceProvider.GetKeyedService(serviceType, serviceKey); + } + + return this._services.GetService(serviceType); + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.DurableTask/DurableAgentJsonUtilities.cs b/dotnet/src/Microsoft.Agents.AI.DurableTask/DurableAgentJsonUtilities.cs new file mode 100644 index 0000000000..e3864e9ad4 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DurableTask/DurableAgentJsonUtilities.cs @@ -0,0 +1,99 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Diagnostics.CodeAnalysis; +using System.Text.Encodings.Web; +using System.Text.Json; +using System.Text.Json.Serialization; +using Microsoft.Agents.AI.DurableTask.State; +using Microsoft.Extensions.AI; + +namespace Microsoft.Agents.AI.DurableTask; + +/// Provides JSON serialization utilities and source-generated contracts for Durable Agent types. +/// +/// +/// This mirrors the pattern used by other libraries (e.g. WorkflowsJsonUtilities) to enable Native AOT and trimming +/// friendly serialization without relying on runtime reflection. It establishes a singleton +/// instance that is preconfigured with: +/// +/// +/// baseline defaults. +/// for default null-value suppression. +/// to tolerate numbers encoded as strings. +/// Chained type info resolvers from shared agent abstractions to cover cross-package types (e.g. , ). +/// +/// +/// Keep the list of [JsonSerializable] types in sync with the Durable Agent data model anytime new state or request/response +/// containers are introduced that must round-trip via JSON. +/// +/// +internal static partial class DurableAgentJsonUtilities +{ + /// + /// Gets the singleton used for Durable Agent serialization. + /// + public static JsonSerializerOptions DefaultOptions { get; } = CreateDefaultOptions(); + + /// + /// Serializes a sequence of chat messages using the durable agent default options. + /// + /// The messages to serialize. + /// A representing the serialized messages. + public static JsonElement Serialize(this IEnumerable messages) => + JsonSerializer.SerializeToElement(messages, DefaultOptions.GetTypeInfo(typeof(IEnumerable))); + + /// + /// Deserializes chat messages from a using durable agent options. + /// + /// The JSON element containing the messages. + /// The deserialized list of chat messages. + public static List DeserializeMessages(this JsonElement element) => + (List?)element.Deserialize(DefaultOptions.GetTypeInfo(typeof(List))) ?? []; + + /// + /// Creates the configured instance for durable agents. + /// + /// The configured options. + [UnconditionalSuppressMessage("ReflectionAnalysis", "IL3050:RequiresDynamicCode", Justification = "Converter is guarded by IsReflectionEnabledByDefault check.")] + [UnconditionalSuppressMessage("Trimming", "IL2026:Members annotated with 'RequiresUnreferencedCodeAttribute' require dynamic access", Justification = "Converter is guarded by IsReflectionEnabledByDefault check.")] + private static JsonSerializerOptions CreateDefaultOptions() + { + // Base configuration from the source-generated context below. + JsonSerializerOptions options = new(JsonContext.Default.Options) + { + Encoder = JavaScriptEncoder.UnsafeRelaxedJsonEscaping, // same as AgentAbstractionsJsonUtilities and AIJsonUtilities + }; + + // Chain in shared abstractions resolver (Microsoft.Extensions.AI + Agent abstractions) so dependent types are covered. + options.TypeInfoResolverChain.Clear(); + options.TypeInfoResolverChain.Add(AgentAbstractionsJsonUtilities.DefaultOptions.TypeInfoResolver!); + options.TypeInfoResolverChain.Add(JsonContext.Default.Options.TypeInfoResolver!); + + if (JsonSerializer.IsReflectionEnabledByDefault) + { + options.Converters.Add(new JsonStringEnumConverter()); + } + + options.MakeReadOnly(); + return options; + } + + // Keep in sync with CreateDefaultOptions above. + [JsonSourceGenerationOptions(JsonSerializerDefaults.Web, + DefaultIgnoreCondition = JsonIgnoreCondition.WhenWritingNull, + NumberHandling = JsonNumberHandling.AllowReadingFromString)] + + // Durable Agent State Types + [JsonSerializable(typeof(DurableAgentState))] + [JsonSerializable(typeof(DurableAgentThread))] + + // Request Types + [JsonSerializable(typeof(RunRequest))] + + // Primitive / Supporting Types + [JsonSerializable(typeof(ChatMessage))] + [JsonSerializable(typeof(JsonElement))] + + [ExcludeFromCodeCoverage] + internal sealed partial class JsonContext : JsonSerializerContext; +} diff --git a/dotnet/src/Microsoft.Agents.AI.DurableTask/DurableAgentRunOptions.cs b/dotnet/src/Microsoft.Agents.AI.DurableTask/DurableAgentRunOptions.cs new file mode 100644 index 0000000000..0f1984ad62 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DurableTask/DurableAgentRunOptions.cs @@ -0,0 +1,36 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Microsoft.Extensions.AI; + +namespace Microsoft.Agents.AI.DurableTask; + +/// +/// Options for running a durable agent. +/// +public sealed class DurableAgentRunOptions : AgentRunOptions +{ + /// + /// Gets or sets whether to enable tool calls for this request. + /// + public bool EnableToolCalls { get; set; } = true; + + /// + /// Gets or sets the collection of tool names to enable. If not specified, all tools are enabled. + /// + public IList? EnableToolNames { get; set; } + + /// + /// Gets or sets the response format for the agent's response. + /// + public ChatResponseFormat? ResponseFormat { get; set; } + + /// + /// Gets or sets whether to fire and forget the agent run request. + /// + /// + /// If is true, the agent run request will be sent and the method will return immediately. + /// The caller will not wait for the agent to complete the run and will not receive a response. This setting is useful for + /// long-running tasks where the caller does not need to wait for the agent to complete the run. + /// + public bool IsFireAndForget { get; set; } +} diff --git a/dotnet/src/Microsoft.Agents.AI.DurableTask/DurableAgentThread.cs b/dotnet/src/Microsoft.Agents.AI.DurableTask/DurableAgentThread.cs new file mode 100644 index 0000000000..32dea2cb18 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DurableTask/DurableAgentThread.cs @@ -0,0 +1,77 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Diagnostics; +using System.Text.Json; +using System.Text.Json.Serialization; + +namespace Microsoft.Agents.AI.DurableTask; + +/// +/// An agent thread implementation for durable agents. +/// +[DebuggerDisplay("{SessionId}")] +public sealed class DurableAgentThread : AgentThread +{ + [JsonConstructor] + internal DurableAgentThread(AgentSessionId sessionId) + { + this.SessionId = sessionId; + } + + /// + /// Gets the agent session ID. + /// + [JsonInclude] + [JsonPropertyName("sessionId")] + internal AgentSessionId SessionId { get; } + + /// + public override JsonElement Serialize(JsonSerializerOptions? jsonSerializerOptions = null) + { + return JsonSerializer.SerializeToElement( + this, + DurableAgentJsonUtilities.DefaultOptions.GetTypeInfo(typeof(DurableAgentThread))); + } + + /// + /// Deserializes a DurableAgentThread from JSON. + /// + /// The serialized thread data. + /// Optional JSON serializer options. + /// The deserialized DurableAgentThread. + internal static DurableAgentThread Deserialize(JsonElement serializedThread, JsonSerializerOptions? jsonSerializerOptions = null) + { + if (!serializedThread.TryGetProperty("sessionId", out JsonElement sessionIdElement) || + sessionIdElement.ValueKind != JsonValueKind.String) + { + throw new JsonException("Invalid or missing sessionId property."); + } + + string sessionIdString = sessionIdElement.GetString() ?? throw new JsonException("sessionId property is null."); + AgentSessionId sessionId = AgentSessionId.Parse(sessionIdString); + return new DurableAgentThread(sessionId); + } + + /// + public override object? GetService(Type serviceType, object? serviceKey = null) + { + // This is a common convention for MAF agents. + if (serviceType == typeof(AgentThreadMetadata)) + { + return new AgentThreadMetadata(conversationId: this.SessionId.ToString()); + } + + if (serviceType == typeof(AgentSessionId)) + { + return this.SessionId; + } + + return base.GetService(serviceType, serviceKey); + } + + /// + public override string ToString() + { + return this.SessionId.ToString(); + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.DurableTask/DurableAgentsOptions.cs b/dotnet/src/Microsoft.Agents.AI.DurableTask/DurableAgentsOptions.cs new file mode 100644 index 0000000000..f2ac3f4c9a --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DurableTask/DurableAgentsOptions.cs @@ -0,0 +1,84 @@ +// Copyright (c) Microsoft. All rights reserved. + +namespace Microsoft.Agents.AI.DurableTask; + +/// +/// Builder for configuring durable agents. +/// +public sealed class DurableAgentsOptions +{ + // Agent names are case-insensitive + private readonly Dictionary> _agentFactories = new(StringComparer.OrdinalIgnoreCase); + + internal DurableAgentsOptions() + { + } + + /// + /// Adds an AI agent factory to the options. + /// + /// The name of the agent. + /// The factory function to create the agent. + /// The options instance. + /// Thrown when or is null. + public DurableAgentsOptions AddAIAgentFactory(string name, Func factory) + { + ArgumentNullException.ThrowIfNull(name); + ArgumentNullException.ThrowIfNull(factory); + this._agentFactories.Add(name, factory); + return this; + } + + /// + /// Adds a list of AI agents to the options. + /// + /// The list of agents to add. + /// The options instance. + /// Thrown when is null. + public DurableAgentsOptions AddAIAgents(params IEnumerable agents) + { + ArgumentNullException.ThrowIfNull(agents); + foreach (AIAgent agent in agents) + { + this.AddAIAgent(agent); + } + + return this; + } + + /// + /// Adds an AI agent to the options. + /// + /// The agent to add. + /// The options instance. + /// Thrown when is null. + /// + /// Thrown when is null or whitespace or when an agent with the same name has already been registered. + /// + public DurableAgentsOptions AddAIAgent(AIAgent agent) + { + ArgumentNullException.ThrowIfNull(agent); + + if (string.IsNullOrWhiteSpace(agent.Name)) + { + throw new ArgumentException($"{nameof(agent.Name)} must not be null or whitespace.", nameof(agent)); + } + + if (this._agentFactories.ContainsKey(agent.Name)) + { + throw new ArgumentException($"An agent with name '{agent.Name}' has already been registered.", nameof(agent)); + } + + this._agentFactories.Add(agent.Name, sp => agent); + return this; + } + + /// + /// Gets the agents that have been added to this builder. + /// + /// A read-only collection of agents. + internal IReadOnlyDictionary> GetAgentFactories() + { + return this._agentFactories.AsReadOnly(); + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.DurableTask/EntityAgentWrapper.cs b/dotnet/src/Microsoft.Agents.AI.DurableTask/EntityAgentWrapper.cs new file mode 100644 index 0000000000..8822ebcc39 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DurableTask/EntityAgentWrapper.cs @@ -0,0 +1,125 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Runtime.CompilerServices; +using Microsoft.Agents.AI; +using Microsoft.DurableTask.Entities; +using Microsoft.Extensions.AI; +using Microsoft.Extensions.DependencyInjection; + +namespace Microsoft.Agents.AI.DurableTask; + +internal sealed class EntityAgentWrapper( + AIAgent innerAgent, + TaskEntityContext entityContext, + RunRequest runRequest, + IServiceProvider? entityScopedServices = null) : DelegatingAIAgent(innerAgent) +{ + private readonly TaskEntityContext _entityContext = entityContext; + private readonly RunRequest _runRequest = runRequest; + private readonly IServiceProvider? _entityScopedServices = entityScopedServices; + + // The ID of the agent is always the entity ID. + protected override string? IdCore => this._entityContext.Id.ToString(); + + public override async Task RunAsync( + IEnumerable messages, + AgentThread? thread = null, + AgentRunOptions? options = null, + CancellationToken cancellationToken = default) + { + AgentRunResponse response = await base.RunAsync( + messages, + thread, + this.GetAgentEntityRunOptions(options), + cancellationToken); + + response.AgentId = this.Id; + return response; + } + + public override async IAsyncEnumerable RunStreamingAsync( + IEnumerable messages, + AgentThread? thread = null, + AgentRunOptions? options = null, + [EnumeratorCancellation] CancellationToken cancellationToken = default) + { + await foreach (AgentRunResponseUpdate update in base.RunStreamingAsync( + messages, + thread, + this.GetAgentEntityRunOptions(options), + cancellationToken)) + { + update.AgentId = this.Id; + yield return update; + } + } + + // Override the GetService method to provide entity-scoped services. + public override object? GetService(Type serviceType, object? serviceKey = null) + { + object? result = null; + if (this._entityScopedServices is not null) + { + result = (serviceKey is not null && this._entityScopedServices is IKeyedServiceProvider keyedServiceProvider) + ? keyedServiceProvider.GetKeyedService(serviceType, serviceKey) + : this._entityScopedServices.GetService(serviceType); + } + + return result ?? base.GetService(serviceType, serviceKey); + } + + private AgentRunOptions GetAgentEntityRunOptions(AgentRunOptions? options = null) + { + // Copied/modified from FunctionInvocationDelegatingAgent.cs in microsoft/agent-framework. + if (options is null || options.GetType() == typeof(AgentRunOptions)) + { + options = new ChatClientAgentRunOptions(); + } + + if (options is not ChatClientAgentRunOptions chatAgentRunOptions) + { + throw new NotSupportedException($"Function Invocation Middleware is only supported without options or with {nameof(ChatClientAgentRunOptions)}."); + } + + Func? originalFactory = chatAgentRunOptions.ChatClientFactory; + + chatAgentRunOptions.ChatClientFactory = chatClient => + { + ChatClientBuilder builder = chatClient.AsBuilder(); + if (originalFactory is not null) + { + builder.Use(originalFactory); + } + + // Update the run options based on the run request. + // NOTE: Function middleware can go here if needed in the future. + return builder.ConfigureOptions( + newOptions => + { + // Update the response format if requested by the caller. + if (this._runRequest.ResponseFormat is not null) + { + newOptions.ResponseFormat = this._runRequest.ResponseFormat; + } + + // Update the tools if requested by the caller. + if (this._runRequest.EnableToolCalls) + { + IList? tools = chatAgentRunOptions.ChatOptions?.Tools; + if (tools is not null && this._runRequest.EnableToolNames?.Count > 0) + { + // Filter tools to only include those with matching names + newOptions.Tools = [.. tools.Where(tool => this._runRequest.EnableToolNames.Contains(tool.Name))]; + } + } + else + { + newOptions.Tools = null; + } + }) + .Build(); + }; + + return options; + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.DurableTask/IAgentResponseHandler.cs b/dotnet/src/Microsoft.Agents.AI.DurableTask/IAgentResponseHandler.cs new file mode 100644 index 0000000000..45a4e9f258 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DurableTask/IAgentResponseHandler.cs @@ -0,0 +1,35 @@ +// Copyright (c) Microsoft. All rights reserved. + +namespace Microsoft.Agents.AI.DurableTask; + +/// +/// Handler for processing responses from the agent. This is typically used to send messages to the user. +/// +public interface IAgentResponseHandler +{ + /// + /// Handles a streaming response update from the agent. This is typically used to send messages to the user. + /// + /// + /// The stream of messages from the agent. + /// + /// + /// Signals that the operation should be cancelled. + /// + ValueTask OnStreamingResponseUpdateAsync( + IAsyncEnumerable messageStream, + CancellationToken cancellationToken); + + /// + /// Handles a discrete response from the agent. This is typically used to send messages to the user. + /// + /// + /// The message from the agent. + /// + /// + /// Signals that the operation should be cancelled. + /// + ValueTask OnAgentResponseAsync( + AgentRunResponse message, + CancellationToken cancellationToken); +} diff --git a/dotnet/src/Microsoft.Agents.AI.DurableTask/IDurableAgentClient.cs b/dotnet/src/Microsoft.Agents.AI.DurableTask/IDurableAgentClient.cs new file mode 100644 index 0000000000..d49999cbbe --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DurableTask/IDurableAgentClient.cs @@ -0,0 +1,21 @@ +// Copyright (c) Microsoft. All rights reserved. + +namespace Microsoft.Agents.AI.DurableTask; + +/// +/// Represents a client for interacting with a durable agent. +/// +internal interface IDurableAgentClient +{ + /// + /// Runs an agent with the specified request. + /// + /// The ID of the target agent session. + /// The request containing the message, role, and configuration. + /// The cancellation token for scheduling the request. + /// A task that returns a handle used to read the agent response. + Task RunAgentAsync( + AgentSessionId sessionId, + RunRequest request, + CancellationToken cancellationToken = default); +} diff --git a/dotnet/src/Microsoft.Agents.AI.DurableTask/Logs.cs b/dotnet/src/Microsoft.Agents.AI.DurableTask/Logs.cs new file mode 100644 index 0000000000..0bec1e149c --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DurableTask/Logs.cs @@ -0,0 +1,49 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Microsoft.Agents.AI.DurableTask; +using Microsoft.Extensions.AI; +using Microsoft.Extensions.Logging; + +internal static partial class Logs +{ + [LoggerMessage( + EventId = 1, + Level = LogLevel.Information, + Message = "[{SessionId}] Request: [{Role}] {Content}")] + public static partial void LogAgentRequest( + this ILogger logger, + AgentSessionId sessionId, + ChatRole role, + string content); + + [LoggerMessage( + EventId = 2, + Level = LogLevel.Information, + Message = "[{SessionId}] Response: [{Role}] {Content} (Input tokens: {InputTokenCount}, Output tokens: {OutputTokenCount}, Total tokens: {TotalTokenCount})")] + public static partial void LogAgentResponse( + this ILogger logger, + AgentSessionId sessionId, + ChatRole role, + string content, + long? inputTokenCount, + long? outputTokenCount, + long? totalTokenCount); + + [LoggerMessage( + EventId = 3, + Level = LogLevel.Information, + Message = "Signalling agent with session ID '{SessionId}'")] + public static partial void LogSignallingAgent(this ILogger logger, AgentSessionId sessionId); + + [LoggerMessage( + EventId = 4, + Level = LogLevel.Information, + Message = "Polling agent with session ID '{SessionId}' for response with correlation ID '{CorrelationId}'")] + public static partial void LogStartPollingForResponse(this ILogger logger, AgentSessionId sessionId, string correlationId); + + [LoggerMessage( + EventId = 5, + Level = LogLevel.Information, + Message = "Found response for agent with session ID '{SessionId}' with correlation ID '{CorrelationId}'")] + public static partial void LogDonePollingForResponse(this ILogger logger, AgentSessionId sessionId, string correlationId); +} diff --git a/dotnet/src/Microsoft.Agents.AI.DurableTask/Microsoft.Agents.AI.DurableTask.csproj b/dotnet/src/Microsoft.Agents.AI.DurableTask/Microsoft.Agents.AI.DurableTask.csproj new file mode 100644 index 0000000000..41284e1085 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DurableTask/Microsoft.Agents.AI.DurableTask.csproj @@ -0,0 +1,39 @@ + + + + $(TargetFrameworksCore) + enable + + + $(NoWarn);CA2007;MEAI001 + + + + + + + Durable Task extensions for Microsoft Agent Framework + Provides distributed durable execution capabilities for agents built with Microsoft Agent Framework. + README.md + + + + + + + + + + + + + + + + + + + + + + diff --git a/dotnet/src/Microsoft.Agents.AI.DurableTask/README.md b/dotnet/src/Microsoft.Agents.AI.DurableTask/README.md new file mode 100644 index 0000000000..85686cce69 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DurableTask/README.md @@ -0,0 +1,42 @@ +# Microsoft.Agents.AI.DurableTask + +The Microsoft Agent Framework provides a programming model for building agents and agent workflows in .NET. This package, the *Durable Task extension for the Agent Framework*, extends the Agent Framework programming model with the following capabilities: + +- Stateful, durable execution of agents in distributed environments +- Automatic conversation history management +- Long-running agent workflows as "durable orchestrator" functions +- Tools and dashboards for managing and monitoring agents and agent workflows + +These capabilities are implemented using foundational technologies from the Durable Task technology stack: + +- [Durable Entities](https://learn.microsoft.com/azure/azure-functions/durable/durable-functions-entities) for stateful, durable execution of agents +- [Durable Orchestrations](https://learn.microsoft.com/azure/azure-functions/durable/durable-functions-orchestrations) for long-running agent workflows +- The [Durable Task Scheduler](https://learn.microsoft.com/azure/azure-functions/durable/durable-task-scheduler/choose-orchestration-framework) for managing durable task execution and observability at scale + +This package can be used by itself or in conjunction with the `Microsoft.Agents.AI.Hosting.AzureFunctions` package, which provides additional features via Azure Functions integration. + +## Install the package + +From the command-line: + +```bash +dotnet add package Microsoft.Agents.AI.DurableTask +``` + +Or directly in your project file: + +```xml + + + +``` + +You can alternatively just reference the `Microsoft.Agents.AI.Hosting.AzureFunctions` package if you're hosting your agents and orchestrations in the Azure Functions .NET Isolated worker. + +## Usage Examples + +For a comprehensive tour of all the functionality, concepts, and APIs, check out the [Azure Functions samples](https://github.com/microsoft/agent-framework/tree/main/dotnet/samples/). + +## Feedback & Contributing + +We welcome feedback and contributions in [our GitHub repo](https://github.com/microsoft/agent-framework). diff --git a/dotnet/src/Microsoft.Agents.AI.DurableTask/RunRequest.cs b/dotnet/src/Microsoft.Agents.AI.DurableTask/RunRequest.cs new file mode 100644 index 0000000000..0fc7ffc7b4 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DurableTask/RunRequest.cs @@ -0,0 +1,83 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; +using Microsoft.Extensions.AI; + +namespace Microsoft.Agents.AI.DurableTask; + +/// +/// Represents a request to run an agent with a specific message and configuration. +/// +public record RunRequest +{ + /// + /// Gets the list of chat messages to send to the agent (for multi-message requests). + /// + public IList Messages { get; init; } = []; + + /// + /// Gets the optional response format for the agent's response. + /// + public ChatResponseFormat? ResponseFormat { get; init; } + + /// + /// Gets whether to enable tool calls for this request. + /// + public bool EnableToolCalls { get; init; } = true; + + /// + /// Gets the collection of tool names to enable. If not specified, all tools are enabled. + /// + public IList? EnableToolNames { get; init; } + + /// + /// Gets or sets the correlation ID for correlating this request with its response. + /// + [JsonInclude] + internal string CorrelationId { get; set; } = Guid.NewGuid().ToString("N"); + + /// + /// Gets or sets the ID of the orchestration that initiated this request (if any). + /// + [JsonInclude] + [JsonIgnore(Condition = JsonIgnoreCondition.WhenWritingNull)] + internal string? OrchestrationId { get; set; } + + /// + /// Initializes a new instance of the class for a single message. + /// + /// The message to send to the agent. + /// The role of the message sender (User or System). + /// Optional response format for the agent's response. + /// Whether to enable tool calls for this request. + /// Optional collection of tool names to enable. If not specified, all tools are enabled. + public RunRequest( + string message, + ChatRole? role = null, + ChatResponseFormat? responseFormat = null, + bool enableToolCalls = true, + IList? enableToolNames = null) + : this([new ChatMessage(role ?? ChatRole.User, message) { CreatedAt = DateTimeOffset.UtcNow }], responseFormat, enableToolCalls, enableToolNames) + { + } + + /// + /// Initializes a new instance of the class for multiple messages. + /// + /// The list of chat messages to send to the agent. + /// Optional response format for the agent's response. + /// Whether to enable tool calls for this request. + /// Optional collection of tool names to enable. If not specified, all tools are enabled. + [JsonConstructor] + public RunRequest( + IList messages, + ChatResponseFormat? responseFormat = null, + bool enableToolCalls = true, + IList? enableToolNames = null) + { + this.Messages = messages; + this.ResponseFormat = responseFormat; + this.EnableToolCalls = enableToolCalls; + this.EnableToolNames = enableToolNames; + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.DurableTask/ServiceCollectionExtensions.cs b/dotnet/src/Microsoft.Agents.AI.DurableTask/ServiceCollectionExtensions.cs new file mode 100644 index 0000000000..2f435e0541 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DurableTask/ServiceCollectionExtensions.cs @@ -0,0 +1,183 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Diagnostics.CodeAnalysis; +using System.Text.Json; +using System.Text.Json.Serialization.Metadata; +using Microsoft.Agents.AI.DurableTask.State; +using Microsoft.DurableTask; +using Microsoft.DurableTask.Client; +using Microsoft.DurableTask.Worker; +using Microsoft.Extensions.DependencyInjection; + +namespace Microsoft.Agents.AI.DurableTask; + +/// +/// Agent-specific extension methods for the class. +/// +public static class ServiceCollectionExtensions +{ + /// + /// Gets a durable agent proxy by name. + /// + /// The service provider. + /// The name of the agent. + /// The durable agent proxy. + /// Thrown if the agent proxy is not found. + public static AIAgent GetDurableAgentProxy(this IServiceProvider services, string name) + { + return services.GetKeyedService(name) + ?? throw new KeyNotFoundException($"A durable agent with name '{name}' has not been registered."); + } + + /// + /// Configures the Durable Agents services via the service collection. + /// + /// The service collection. + /// A delegate to configure the durable agents. + /// A delegate to configure the Durable Task worker. + /// A delegate to configure the Durable Task client. + /// The service collection. + public static IServiceCollection ConfigureDurableAgents( + this IServiceCollection services, + Action configure, + Action? workerBuilder = null, + Action? clientBuilder = null) + { + ArgumentNullException.ThrowIfNull(configure); + + DurableAgentsOptions options = services.ConfigureDurableAgents(configure); + + // A worker is required to run the agent entities + services.AddDurableTaskWorker(builder => + { + workerBuilder?.Invoke(builder); + + builder.AddTasks(registry => + { + foreach (string name in options.GetAgentFactories().Keys) + { + registry.AddEntity(AgentSessionId.ToEntityName(name)); + } + }); + }); + + // The client is needed to send notifications to the agent entities from non-orchestrator code + if (clientBuilder != null) + { + services.AddDurableTaskClient(clientBuilder); + } + + services.AddSingleton(); + + return services; + } + + // This is internal because it's also used by Microsoft.Azure.Functions.DurableAgents, which is a friend assembly project. + internal static DurableAgentsOptions ConfigureDurableAgents( + this IServiceCollection services, + Action configure) + { + DurableAgentsOptions options = new(); + configure(options); + + IReadOnlyDictionary> agents = options.GetAgentFactories(); + + // The agent dictionary contains the real agent factories, which is used by the agent entities. + services.AddSingleton(agents); + + // The keyed services are used to resolve durable agent *proxy* instances for external clients. + foreach (var factory in agents) + { + services.AddKeyedSingleton(factory.Key, (sp, _) => factory.Value(sp).AsDurableAgentProxy(sp)); + } + + // A custom data converter is needed because the default chat client uses camel case for JSON properties, + // which is not the default behavior for the Durable Task SDK. + services.AddSingleton(); + + return options; + } + + /// + /// Validates that an agent with the specified name has been registered. + /// + /// The service provider. + /// The name of the agent to validate. + /// + /// Thrown when the agent dictionary is not registered in the service provider. + /// + /// + /// Thrown when the agent with the specified name has not been registered. + /// + internal static void ValidateAgentIsRegistered(IServiceProvider services, string agentName) + { + IReadOnlyDictionary>? agents = + services.GetService>>() + ?? throw new InvalidOperationException( + $"Durable agents have not been configured. Ensure {nameof(ConfigureDurableAgents)} has been called on the service collection."); + + if (!agents.ContainsKey(agentName)) + { + throw new AgentNotRegisteredException(agentName); + } + } + + private sealed class DefaultDataConverter : DataConverter + { + // Use durable agent options (web defaults + camel case by default) with case-insensitive matching. + // We clone to apply naming/casing tweaks while retaining source-generated metadata where available. + private static readonly JsonSerializerOptions s_options = new(DurableAgentJsonUtilities.DefaultOptions) + { + PropertyNamingPolicy = JsonNamingPolicy.CamelCase, + PropertyNameCaseInsensitive = true, + }; + + [UnconditionalSuppressMessage("Trimming", "IL2026", Justification = "Fallback path uses reflection when metadata unavailable.")] + [UnconditionalSuppressMessage("ReflectionAnalysis", "IL3050", Justification = "Fallback path uses reflection when metadata unavailable.")] + public override object? Deserialize(string? data, Type targetType) + { + if (data is null) + { + return null; + } + + if (targetType == typeof(DurableAgentState)) + { + return JsonSerializer.Deserialize(data, DurableAgentStateJsonContext.Default.DurableAgentState); + } + + JsonTypeInfo? typeInfo = s_options.GetTypeInfo(targetType); + if (typeInfo is JsonTypeInfo typedInfo) + { + return JsonSerializer.Deserialize(data, typedInfo); + } + + // Fallback (may trigger trimming/AOT warnings for unsupported dynamic types). + return JsonSerializer.Deserialize(data, targetType, s_options); + } + + [return: NotNullIfNotNull(nameof(value))] + [UnconditionalSuppressMessage("Trimming", "IL2026", Justification = "Fallback path uses reflection when metadata unavailable.")] + [UnconditionalSuppressMessage("ReflectionAnalysis", "IL3050", Justification = "Fallback path uses reflection when metadata unavailable.")] + public override string? Serialize(object? value) + { + if (value is null) + { + return null; + } + + if (value is DurableAgentState durableAgentState) + { + return JsonSerializer.Serialize(durableAgentState, DurableAgentStateJsonContext.Default.DurableAgentState); + } + + JsonTypeInfo? typeInfo = s_options.GetTypeInfo(value.GetType()); + if (typeInfo is JsonTypeInfo typedInfo) + { + return JsonSerializer.Serialize(value, typedInfo); + } + + return JsonSerializer.Serialize(value, s_options); + } + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentState.cs b/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentState.cs new file mode 100644 index 0000000000..35aef33544 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentState.cs @@ -0,0 +1,27 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +namespace Microsoft.Agents.AI.DurableTask.State; + +/// +/// Represents the state of a durable agent, including its conversation history. +/// +[JsonConverter(typeof(DurableAgentStateJsonConverter))] +internal sealed class DurableAgentState +{ + /// + /// Gets the data of the durable agent. + /// + [JsonPropertyName("data")] + public DurableAgentStateData Data { get; init; } = new(); + + /// + /// Gets the schema version of the durable agent state. + /// + /// + /// The version is specified in semver (i.e. "major.minor.patch") format. + /// + [JsonPropertyName("schemaVersion")] + public string SchemaVersion { get; init; } = "1.1.0"; +} diff --git a/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateContent.cs b/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateContent.cs new file mode 100644 index 0000000000..62f9f18d60 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateContent.cs @@ -0,0 +1,60 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json; +using System.Text.Json.Serialization; +using Microsoft.Extensions.AI; + +namespace Microsoft.Agents.AI.DurableTask.State; + +/// +/// Base class for durable agent state content types. +/// +[JsonPolymorphic(TypeDiscriminatorPropertyName = "$type")] +[JsonDerivedType(typeof(DurableAgentStateDataContent), "data")] +[JsonDerivedType(typeof(DurableAgentStateErrorContent), "error")] +[JsonDerivedType(typeof(DurableAgentStateFunctionCallContent), "functionCall")] +[JsonDerivedType(typeof(DurableAgentStateFunctionResultContent), "functionResult")] +[JsonDerivedType(typeof(DurableAgentStateHostedFileContent), "hostedFile")] +[JsonDerivedType(typeof(DurableAgentStateHostedVectorStoreContent), "hostedVectorStore")] +[JsonDerivedType(typeof(DurableAgentStateTextContent), "text")] +[JsonDerivedType(typeof(DurableAgentStateTextReasoningContent), "reasoning")] +[JsonDerivedType(typeof(DurableAgentStateUriContent), "uri")] +[JsonDerivedType(typeof(DurableAgentStateUsageContent), "usage")] +[JsonDerivedType(typeof(DurableAgentStateUnknownContent), "unknown")] +internal abstract class DurableAgentStateContent +{ + /// + /// Gets any additional data found during deserialization that does not map to known properties. + /// + [JsonExtensionData] + public IDictionary? ExtensionData { get; set; } + + /// + /// Converts this durable agent state content to an . + /// + /// A converted instance. + public abstract AIContent ToAIContent(); + + /// + /// Creates a from an . + /// + /// The to convert. + /// A representing the original . + public static DurableAgentStateContent FromAIContent(AIContent content) + { + return content switch + { + DataContent dataContent => DurableAgentStateDataContent.FromDataContent(dataContent), + ErrorContent errorContent => DurableAgentStateErrorContent.FromErrorContent(errorContent), + FunctionCallContent functionCallContent => DurableAgentStateFunctionCallContent.FromFunctionCallContent(functionCallContent), + FunctionResultContent functionResultContent => DurableAgentStateFunctionResultContent.FromFunctionResultContent(functionResultContent), + HostedFileContent hostedFileContent => DurableAgentStateHostedFileContent.FromHostedFileContent(hostedFileContent), + HostedVectorStoreContent hostedVectorStoreContent => DurableAgentStateHostedVectorStoreContent.FromHostedVectorStoreContent(hostedVectorStoreContent), + TextContent textContent => DurableAgentStateTextContent.FromTextContent(textContent), + TextReasoningContent textReasoningContent => DurableAgentStateTextReasoningContent.FromTextReasoningContent(textReasoningContent), + UriContent uriContent => DurableAgentStateUriContent.FromUriContent(uriContent), + UsageContent usageContent => DurableAgentStateUsageContent.FromUsageContent(usageContent), + _ => DurableAgentStateUnknownContent.FromUnknownContent(content) + }; + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateData.cs b/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateData.cs new file mode 100644 index 0000000000..f51820dcf5 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateData.cs @@ -0,0 +1,25 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json; +using System.Text.Json.Serialization; + +namespace Microsoft.Agents.AI.DurableTask.State; + +/// +/// Represents the data of a durable agent, including its conversation history. +/// +internal sealed class DurableAgentStateData +{ + /// + /// Gets the ordered list of state entries representing the complete conversation history. + /// This includes both user messages and agent responses in chronological order. + /// + [JsonPropertyName("conversationHistory")] + public IList ConversationHistory { get; init; } = []; + + /// + /// Gets any additional data found during deserialization that does not map to known properties. + /// + [JsonExtensionData] + public IDictionary? ExtensionData { get; set; } +} diff --git a/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateDataContent.cs b/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateDataContent.cs new file mode 100644 index 0000000000..9954213bd7 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateDataContent.cs @@ -0,0 +1,45 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; +using Microsoft.Extensions.AI; + +namespace Microsoft.Agents.AI.DurableTask.State; + +/// +/// Represents a durable agent state content that contains data content. +/// +internal sealed class DurableAgentStateDataContent : DurableAgentStateContent +{ + /// + /// Gets the URI of the data content. + /// + [JsonPropertyName("uri")] + public required string Uri { get; init; } + + /// + /// Gets the media type of the data content. + /// + [JsonPropertyName("mediaType")] + [JsonIgnore(Condition = JsonIgnoreCondition.WhenWritingNull)] + public string? MediaType { get; init; } + + /// + /// Creates a from a . + /// + /// The to convert. + /// A representing the original . + public static DurableAgentStateDataContent FromDataContent(DataContent content) + { + return new DurableAgentStateDataContent() + { + MediaType = content.MediaType, + Uri = content.Uri + }; + } + + /// + public override AIContent ToAIContent() + { + return new DataContent(this.Uri, this.MediaType); + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateEntry.cs b/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateEntry.cs new file mode 100644 index 0000000000..2f04c90097 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateEntry.cs @@ -0,0 +1,44 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json; +using System.Text.Json.Serialization; + +namespace Microsoft.Agents.AI.DurableTask.State; + +/// +/// Represents a single entry in the durable agent state, which can either be a +/// user/system request or agent response. +/// +[JsonPolymorphic(TypeDiscriminatorPropertyName = "$type")] +[JsonDerivedType(typeof(DurableAgentStateRequest), "request")] +[JsonDerivedType(typeof(DurableAgentStateResponse), "response")] +internal abstract class DurableAgentStateEntry +{ + /// + /// Gets the correlation ID for this entry. + /// + /// + /// This ID is used to correlate back to its + /// . + /// + [JsonPropertyName("correlationId")] + public required string CorrelationId { get; init; } + + /// + /// Gets the timestamp when this entry was created. + /// + [JsonPropertyName("createdAt")] + public required DateTimeOffset CreatedAt { get; init; } + + /// + /// Gets the list of messages associated with this entry, in chronological order. + /// + [JsonPropertyName("messages")] + public IReadOnlyList Messages { get; init; } = []; + + /// + /// Gets any additional data found during deserialization that does not map to known properties. + /// + [JsonExtensionData] + public IDictionary? ExtensionData { get; set; } +} diff --git a/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateErrorContent.cs b/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateErrorContent.cs new file mode 100644 index 0000000000..17e5fea75f --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateErrorContent.cs @@ -0,0 +1,59 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; +using Microsoft.Extensions.AI; + +namespace Microsoft.Agents.AI.DurableTask.State; + +/// +/// Represents durable agent state content that contains error content. +/// +internal sealed class DurableAgentStateErrorContent : DurableAgentStateContent +{ + /// + /// Gets the error message. + /// + [JsonPropertyName("message")] + [JsonIgnore(Condition = JsonIgnoreCondition.WhenWritingNull)] + public string? Message { get; init; } + + /// + /// Gets the error code. + /// + [JsonPropertyName("errorCode")] + [JsonIgnore(Condition = JsonIgnoreCondition.WhenWritingNull)] + public string? ErrorCode { get; init; } + + /// + /// Gets the error details. + /// + [JsonPropertyName("details")] + [JsonIgnore(Condition = JsonIgnoreCondition.WhenWritingNull)] + public string? Details { get; init; } + + /// + /// Creates a from an . + /// + /// The to convert. + /// A representing the original + /// . + public static DurableAgentStateErrorContent FromErrorContent(ErrorContent content) + { + return new DurableAgentStateErrorContent() + { + Details = content.Details, + ErrorCode = content.ErrorCode, + Message = content.Message + }; + } + + /// + public override AIContent ToAIContent() + { + return new ErrorContent(this.Message) + { + Details = this.Details, + ErrorCode = this.ErrorCode + }; + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateFunctionCallContent.cs b/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateFunctionCallContent.cs new file mode 100644 index 0000000000..1deccc8a77 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateFunctionCallContent.cs @@ -0,0 +1,63 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Collections.Immutable; +using System.Text.Json.Serialization; +using Microsoft.Extensions.AI; + +namespace Microsoft.Agents.AI.DurableTask.State; + +/// +/// Durable agent state content representing a function call. +/// +internal sealed class DurableAgentStateFunctionCallContent : DurableAgentStateContent +{ + /// + /// The function call arguments. + /// + /// TODO: Consider ensuring that empty dictionaries are omitted from serialization. + [JsonPropertyName("arguments")] + public required IReadOnlyDictionary Arguments { get; init; } = + ImmutableDictionary.Empty; + + /// + /// Gets the function call identifier. + /// + /// + /// This is used to correlate this function call with its resulting + /// . + /// + [JsonPropertyName("callId")] + public required string CallId { get; init; } + + /// + /// Gets the function name. + /// + [JsonPropertyName("name")] + public required string Name { get; init; } + + /// + /// Creates a from a . + /// + /// The to convert. + /// + /// A representing the original content. + /// + public static DurableAgentStateFunctionCallContent FromFunctionCallContent(FunctionCallContent content) + { + return new DurableAgentStateFunctionCallContent() + { + Arguments = content.Arguments?.ToDictionary() ?? [], + CallId = content.CallId, + Name = content.Name + }; + } + + /// + public override AIContent ToAIContent() + { + return new FunctionCallContent( + this.CallId, + this.Name, + new Dictionary(this.Arguments)); + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateFunctionResultContent.cs b/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateFunctionResultContent.cs new file mode 100644 index 0000000000..9237fdfa76 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateFunctionResultContent.cs @@ -0,0 +1,49 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; +using Microsoft.Extensions.AI; + +namespace Microsoft.Agents.AI.DurableTask.State; + +/// +/// Represents the function result content for a durable agent state response. +/// +internal sealed class DurableAgentStateFunctionResultContent : DurableAgentStateContent +{ + /// + /// Gets the function call identifier. + /// + /// + /// This is used to correlate this function result with its originating + /// . + /// + [JsonPropertyName("callId")] + public required string CallId { get; init; } + + /// + /// Gets the function result. + /// + [JsonPropertyName("result")] + [JsonIgnore(Condition = JsonIgnoreCondition.WhenWritingNull)] + public object? Result { get; init; } + + /// + /// Creates a from a . + /// + /// The to convert. + /// A representing the original content. + public static DurableAgentStateFunctionResultContent FromFunctionResultContent(FunctionResultContent content) + { + return new DurableAgentStateFunctionResultContent() + { + CallId = content.CallId, + Result = content.Result + }; + } + + /// + public override AIContent ToAIContent() + { + return new FunctionResultContent(this.CallId, this.Result); + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateHostedFileContent.cs b/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateHostedFileContent.cs new file mode 100644 index 0000000000..c6fc860ac0 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateHostedFileContent.cs @@ -0,0 +1,39 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; +using Microsoft.Extensions.AI; + +namespace Microsoft.Agents.AI.DurableTask.State; + +/// +/// Represents durable agent state content that contains hosted file content. +/// +internal sealed class DurableAgentStateHostedFileContent : DurableAgentStateContent +{ + /// + /// Gets the file ID of the hosted file content. + /// + [JsonPropertyName("fileId")] + public required string FileId { get; init; } + + /// + /// Creates a from a . + /// + /// The to convert. + /// + /// A representing the original . + /// + public static DurableAgentStateHostedFileContent FromHostedFileContent(HostedFileContent content) + { + return new DurableAgentStateHostedFileContent() + { + FileId = content.FileId + }; + } + + /// + public override AIContent ToAIContent() + { + return new HostedFileContent(this.FileId); + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateHostedVectorStoreContent.cs b/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateHostedVectorStoreContent.cs new file mode 100644 index 0000000000..f7b615564b --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateHostedVectorStoreContent.cs @@ -0,0 +1,39 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; +using Microsoft.Extensions.AI; + +namespace Microsoft.Agents.AI.DurableTask.State; + +/// +/// Represents durable agent state content that contains hosted vector store content. +/// +internal sealed class DurableAgentStateHostedVectorStoreContent : DurableAgentStateContent +{ + /// + /// Gets the vector store ID of the hosted vector store content. + /// + [JsonPropertyName("vectorStoreId")] + public required string VectorStoreId { get; init; } + + /// + /// Creates a from a . + /// + /// The to convert. + /// + /// A representing the original . + /// + public static DurableAgentStateHostedVectorStoreContent FromHostedVectorStoreContent(HostedVectorStoreContent content) + { + return new DurableAgentStateHostedVectorStoreContent() + { + VectorStoreId = content.VectorStoreId + }; + } + + /// + public override AIContent ToAIContent() + { + return new HostedVectorStoreContent(this.VectorStoreId); + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateJsonContext.cs b/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateJsonContext.cs new file mode 100644 index 0000000000..4ad9a62835 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateJsonContext.cs @@ -0,0 +1,40 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json; +using System.Text.Json.Nodes; +using System.Text.Json.Serialization; + +namespace Microsoft.Agents.AI.DurableTask.State; + +[JsonSourceGenerationOptions(WriteIndented = false)] +[JsonSerializable(typeof(DurableAgentState))] +[JsonSerializable(typeof(DurableAgentStateContent))] +[JsonSerializable(typeof(DurableAgentStateData))] +[JsonSerializable(typeof(DurableAgentStateEntry))] +[JsonSerializable(typeof(DurableAgentStateMessage))] +// Function call and result content +[JsonSerializable(typeof(Dictionary))] +[JsonSerializable(typeof(IDictionary))] +[JsonSerializable(typeof(JsonDocument))] +[JsonSerializable(typeof(JsonElement))] +[JsonSerializable(typeof(JsonNode))] +[JsonSerializable(typeof(JsonObject))] +[JsonSerializable(typeof(JsonValue))] +[JsonSerializable(typeof(JsonArray))] +[JsonSerializable(typeof(IEnumerable))] +[JsonSerializable(typeof(char))] +[JsonSerializable(typeof(string))] +[JsonSerializable(typeof(int))] +[JsonSerializable(typeof(short))] +[JsonSerializable(typeof(long))] +[JsonSerializable(typeof(uint))] +[JsonSerializable(typeof(ushort))] +[JsonSerializable(typeof(ulong))] +[JsonSerializable(typeof(float))] +[JsonSerializable(typeof(double))] +[JsonSerializable(typeof(decimal))] +[JsonSerializable(typeof(bool))] +[JsonSerializable(typeof(TimeSpan))] +[JsonSerializable(typeof(DateTime))] +[JsonSerializable(typeof(DateTimeOffset))] +internal sealed partial class DurableAgentStateJsonContext : JsonSerializerContext; diff --git a/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateJsonConverter.cs b/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateJsonConverter.cs new file mode 100644 index 0000000000..4c7796b36c --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateJsonConverter.cs @@ -0,0 +1,71 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json; +using System.Text.Json.Serialization; + +namespace Microsoft.Agents.AI.DurableTask.State; + +/// +/// JSON converter for which performs schema version checks before deserialization. +/// +internal sealed class DurableAgentStateJsonConverter : JsonConverter +{ + private const string SchemaVersionPropertyName = "schemaVersion"; + private const string DataPropertyName = "data"; + + /// + public override DurableAgentState? Read(ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options) + { + JsonElement? element = JsonSerializer.Deserialize( + ref reader, + DurableAgentStateJsonContext.Default.JsonElement); + + if (element is null) + { + throw new JsonException("The durable agent state is not valid JSON."); + } + + if (!element.Value.TryGetProperty(SchemaVersionPropertyName, out JsonElement versionElement)) + { + throw new InvalidOperationException("The durable agent state is missing the 'schemaVersion' property."); + } + + if (!Version.TryParse(versionElement.GetString(), out Version? schemaVersion)) + { + throw new InvalidOperationException("The durable agent state has an invalid 'schemaVersion' property."); + } + + if (schemaVersion.Major != 1) + { + throw new InvalidOperationException($"The durable agent state schema version '{schemaVersion}' is not supported."); + } + + if (!element.Value.TryGetProperty(DataPropertyName, out JsonElement dataElement)) + { + throw new InvalidOperationException("The durable agent state is missing the 'data' property."); + } + + DurableAgentStateData? data = dataElement.Deserialize( + DurableAgentStateJsonContext.Default.DurableAgentStateData); + + return new DurableAgentState + { + SchemaVersion = schemaVersion.ToString(), + Data = data ?? new DurableAgentStateData() + }; + } + + /// + public override void Write(Utf8JsonWriter writer, DurableAgentState value, JsonSerializerOptions options) + { + writer.WriteStartObject(); + writer.WritePropertyName(SchemaVersionPropertyName); + writer.WriteStringValue(value.SchemaVersion); + writer.WritePropertyName(DataPropertyName); + JsonSerializer.Serialize( + writer, + value.Data, + DurableAgentStateJsonContext.Default.DurableAgentStateData); + writer.WriteEndObject(); + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateMessage.cs b/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateMessage.cs new file mode 100644 index 0000000000..294453c149 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateMessage.cs @@ -0,0 +1,76 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json; +using System.Text.Json.Serialization; +using Microsoft.Extensions.AI; + +namespace Microsoft.Agents.AI.DurableTask.State; + +/// +/// Represents a single message within a durable agent state entry. +/// +internal sealed class DurableAgentStateMessage +{ + /// + /// Gets the name of the author of this message. + /// + [JsonPropertyName("authorName")] + [JsonIgnore(Condition = JsonIgnoreCondition.WhenWritingNull)] + public string? AuthorName { get; init; } + + /// + /// Gets the timestamp when this message was created. + /// + [JsonPropertyName("createdAt")] + [JsonIgnore(Condition = JsonIgnoreCondition.WhenWritingNull)] + public DateTimeOffset? CreatedAt { get; init; } + + /// + /// Gets the contents of this message. + /// + [JsonPropertyName("contents")] + public IReadOnlyList Contents { get; init; } = []; + + /// + /// Gets the role of the message sender (e.g., "user", "assistant", "system"). + /// + [JsonPropertyName("role")] + public required string Role { get; init; } + + /// + /// Gets any additional data found during deserialization that does not map to known properties. + /// + [JsonExtensionData] + public IDictionary? ExtensionData { get; set; } + + /// + /// Creates a from a . + /// + /// The to convert. + /// A representing the original message. + public static DurableAgentStateMessage FromChatMessage(ChatMessage message) + { + return new DurableAgentStateMessage() + { + CreatedAt = message.CreatedAt, + AuthorName = message.AuthorName, + Role = message.Role.ToString(), + Contents = message.Contents.Select(DurableAgentStateContent.FromAIContent).ToList() + }; + } + + /// + /// Converts this to a . + /// + /// A representing this message. + public ChatMessage ToChatMessage() + { + return new ChatMessage() + { + CreatedAt = this.CreatedAt, + AuthorName = this.AuthorName, + Contents = this.Contents.Select(c => c.ToAIContent()).ToList(), + Role = new(this.Role) + }; + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateRequest.cs b/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateRequest.cs new file mode 100644 index 0000000000..6349b97c61 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateRequest.cs @@ -0,0 +1,57 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json; +using System.Text.Json.Serialization; +using Microsoft.Extensions.AI; + +namespace Microsoft.Agents.AI.DurableTask.State; + +/// +/// Represents a user or system request entry in the durable agent state. +/// +internal sealed class DurableAgentStateRequest : DurableAgentStateEntry +{ + /// + /// Gets the ID of the orchestration that initiated this request (if any). + /// + [JsonPropertyName("orchestrationId")] + public string? OrchestrationId { get; init; } + + /// + /// Gets the expected response type for this request (e.g. "json" or "text"). + /// + /// + /// If omitted, the expectation is that the agent will respond in plain text. + /// + [JsonPropertyName("responseType")] + public string? ResponseType { get; init; } + + /// + /// Gets the expected response JSON schema for this request, if applicable. + /// + /// + /// This is only applicable when is "json". + /// If omitted, no specific schema is expected. + /// + [JsonPropertyName("responseSchema")] + [JsonIgnore(Condition = JsonIgnoreCondition.WhenWritingNull)] + public JsonElement? ResponseSchema { get; init; } + + /// + /// Creates a from a . + /// + /// The to convert. + /// A representing the original request. + public static DurableAgentStateRequest FromRunRequest(RunRequest request) + { + return new DurableAgentStateRequest() + { + CorrelationId = request.CorrelationId, + OrchestrationId = request.OrchestrationId, + Messages = request.Messages.Select(DurableAgentStateMessage.FromChatMessage).ToList(), + CreatedAt = request.Messages.Min(m => m.CreatedAt) ?? DateTimeOffset.UtcNow, + ResponseType = request.ResponseFormat is ChatResponseFormatJson ? "json" : "text", + ResponseSchema = (request.ResponseFormat as ChatResponseFormatJson)?.Schema + }; + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateResponse.cs b/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateResponse.cs new file mode 100644 index 0000000000..216bb6e05c --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateResponse.cs @@ -0,0 +1,49 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +namespace Microsoft.Agents.AI.DurableTask.State; + +/// +/// Represents a durable agent state entry that is a response from the agent. +/// +internal sealed class DurableAgentStateResponse : DurableAgentStateEntry +{ + /// + /// Gets the usage details for this state response. + /// + [JsonPropertyName("usage")] + [JsonIgnore(Condition = JsonIgnoreCondition.WhenWritingNull)] + public DurableAgentStateUsage? Usage { get; init; } + + /// + /// Creates a from an . + /// + /// The correlation ID linking this response to its request. + /// The to convert. + /// A representing the original response. + public static DurableAgentStateResponse FromRunResponse(string correlationId, AgentRunResponse response) + { + return new DurableAgentStateResponse() + { + CorrelationId = correlationId, + CreatedAt = response.CreatedAt ?? response.Messages.Max(m => m.CreatedAt) ?? DateTimeOffset.UtcNow, + Messages = response.Messages.Select(DurableAgentStateMessage.FromChatMessage).ToList(), + Usage = DurableAgentStateUsage.FromUsage(response.Usage) + }; + } + + /// + /// Converts this back to an . + /// + /// A representing this response. + public AgentRunResponse ToRunResponse() + { + return new AgentRunResponse() + { + CreatedAt = this.CreatedAt, + Messages = this.Messages.Select(m => m.ToChatMessage()).ToList(), + Usage = this.Usage?.ToUsageDetails(), + }; + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateTextContent.cs b/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateTextContent.cs new file mode 100644 index 0000000000..0f3085465a --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateTextContent.cs @@ -0,0 +1,38 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; +using Microsoft.Extensions.AI; + +namespace Microsoft.Agents.AI.DurableTask.State; + +/// +/// Represents the text content for a durable agent state entry. +/// +internal sealed class DurableAgentStateTextContent : DurableAgentStateContent +{ + /// + /// Gets the text message content. + /// + [JsonPropertyName("text")] + [JsonIgnore(Condition = JsonIgnoreCondition.WhenWritingNull)] + public required string? Text { get; init; } + + /// + /// Creates a from a . + /// + /// The to convert. + /// A representing the original content. + public static DurableAgentStateTextContent FromTextContent(TextContent content) + { + return new DurableAgentStateTextContent() + { + Text = content.Text + }; + } + + /// + public override AIContent ToAIContent() + { + return new TextContent(this.Text); + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateTextReasoningContent.cs b/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateTextReasoningContent.cs new file mode 100644 index 0000000000..9b5d6eba34 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateTextReasoningContent.cs @@ -0,0 +1,38 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; +using Microsoft.Extensions.AI; + +namespace Microsoft.Agents.AI.DurableTask.State; + +/// +/// Represents the text reasoning content for a durable agent state entry. +/// +internal sealed class DurableAgentStateTextReasoningContent : DurableAgentStateContent +{ + /// + /// Gets the text reasoning content. + /// + [JsonPropertyName("text")] + [JsonIgnore(Condition = JsonIgnoreCondition.WhenWritingNull)] + public string? Text { get; init; } + + /// + /// Creates a from a . + /// + /// The to convert. + /// A representing the original content. + public static DurableAgentStateTextReasoningContent FromTextReasoningContent(TextReasoningContent content) + { + return new DurableAgentStateTextReasoningContent() + { + Text = content.Text + }; + } + + /// + public override AIContent ToAIContent() + { + return new TextReasoningContent(this.Text); + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateUnknownContent.cs b/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateUnknownContent.cs new file mode 100644 index 0000000000..00a180bba3 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateUnknownContent.cs @@ -0,0 +1,43 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json; +using System.Text.Json.Serialization; +using Microsoft.Extensions.AI; + +namespace Microsoft.Agents.AI.DurableTask.State; + +/// +/// Represents the unknown content for a durable agent state entry. +/// +internal sealed class DurableAgentStateUnknownContent : DurableAgentStateContent +{ + /// + /// Gets the serialized unknown content. + /// + [JsonPropertyName("content")] + public required JsonElement Content { get; init; } + + /// + /// Creates a from an . + /// + /// The to convert. + /// A representing the original content. + public static DurableAgentStateUnknownContent FromUnknownContent(AIContent content) + { + return new DurableAgentStateUnknownContent() + { + Content = JsonSerializer.SerializeToElement( + value: content, + jsonTypeInfo: AIJsonUtilities.DefaultOptions.GetTypeInfo(typeof(AIContent))) + }; + } + + /// + public override AIContent ToAIContent() + { + AIContent? content = this.Content.Deserialize( + jsonTypeInfo: AIJsonUtilities.DefaultOptions.GetTypeInfo(typeof(AIContent))) as AIContent; + + return content ?? throw new InvalidOperationException($"The content '{this.Content}' is not valid AI content."); + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateUriContent.cs b/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateUriContent.cs new file mode 100644 index 0000000000..8c6bbb8f24 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateUriContent.cs @@ -0,0 +1,44 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; +using Microsoft.Extensions.AI; + +namespace Microsoft.Agents.AI.DurableTask.State; + +/// +/// Represents URI content for a durable agent state message. +/// +internal sealed class DurableAgentStateUriContent : DurableAgentStateContent +{ + /// + /// Gets the URI of the content. + /// + [JsonPropertyName("uri")] + public required Uri Uri { get; init; } + + /// + /// Gets the media type of the content. + /// + [JsonPropertyName("mediaType")] + public required string MediaType { get; init; } + + /// + /// Creates a from a . + /// + /// The to convert. + /// A representing the original content. + public static DurableAgentStateUriContent FromUriContent(UriContent uriContent) + { + return new DurableAgentStateUriContent() + { + MediaType = uriContent.MediaType, + Uri = uriContent.Uri + }; + } + + /// + public override AIContent ToAIContent() + { + return new UriContent(this.Uri, this.MediaType); + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateUsage.cs b/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateUsage.cs new file mode 100644 index 0000000000..1b3714faca --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateUsage.cs @@ -0,0 +1,71 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Diagnostics.CodeAnalysis; +using System.Text.Json; +using System.Text.Json.Serialization; +using Microsoft.Extensions.AI; + +namespace Microsoft.Agents.AI.DurableTask.State; + +/// +/// Represents the token usage details for a durable agent state response. +/// +internal sealed class DurableAgentStateUsage +{ + /// + /// Gets the number of input tokens used. + /// + [JsonPropertyName("inputTokenCount")] + [JsonIgnore(Condition = JsonIgnoreCondition.WhenWritingNull)] + public long? InputTokenCount { get; init; } + + /// + /// Gets the number of output tokens used. + /// + [JsonPropertyName("outputTokenCount")] + [JsonIgnore(Condition = JsonIgnoreCondition.WhenWritingNull)] + public long? OutputTokenCount { get; init; } + + /// + /// Gets the total number of tokens used. + /// + [JsonPropertyName("totalTokenCount")] + [JsonIgnore(Condition = JsonIgnoreCondition.WhenWritingNull)] + public long? TotalTokenCount { get; init; } + + /// + /// Gets any additional data found during deserialization that does not map to known properties. + /// + [JsonExtensionData] + public IDictionary? ExtensionData { get; set; } + + /// + /// Creates a from a . + /// + /// The to convert. + /// A representing the original usage details. + [return: NotNullIfNotNull(nameof(usage))] + public static DurableAgentStateUsage? FromUsage(UsageDetails? usage) => + usage is not null + ? new() + { + InputTokenCount = usage.InputTokenCount, + OutputTokenCount = usage.OutputTokenCount, + TotalTokenCount = usage.TotalTokenCount + } + : null; + + /// + /// Converts this back to a . + /// + /// A representing this usage. + public UsageDetails ToUsageDetails() + { + return new() + { + InputTokenCount = this.InputTokenCount, + OutputTokenCount = this.OutputTokenCount, + TotalTokenCount = this.TotalTokenCount + }; + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateUsageContent.cs b/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateUsageContent.cs new file mode 100644 index 0000000000..bdad860e62 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DurableTask/State/DurableAgentStateUsageContent.cs @@ -0,0 +1,37 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; +using Microsoft.Extensions.AI; + +namespace Microsoft.Agents.AI.DurableTask.State; + +/// +/// Represents the content for a durable agent state message. +/// +internal sealed class DurableAgentStateUsageContent : DurableAgentStateContent +{ + /// + /// Gets the usage details. + /// + [JsonPropertyName("usage")] + public DurableAgentStateUsage Usage { get; init; } = new(); + + /// + /// Creates a from a . + /// + /// The to convert. + /// A representing the original content. + public static DurableAgentStateUsageContent FromUsageContent(UsageContent content) + { + return new DurableAgentStateUsageContent() + { + Usage = DurableAgentStateUsage.FromUsage(content.Details) + }; + } + + /// + public override AIContent ToAIContent() + { + return new UsageContent(this.Usage.ToUsageDetails()); + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.DurableTask/State/README.md b/dotnet/src/Microsoft.Agents.AI.DurableTask/State/README.md new file mode 100644 index 0000000000..09bb13c51e --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DurableTask/State/README.md @@ -0,0 +1,147 @@ +# Durable Agent State + +Durable agents are represented as durable entities, with each session (i.e. thread) of conversation history stored as JSON-serialized state for an individual entity instance. + +## State Schema + +The [schema](../../../../schemas/durable-agent-entity-state.json) for durable agent state is a distillation of the prompt and response messages accumulated over the lifetime of a session. While these messages and content originate from Microsoft Agent Framework types (for .NET, see [ChatMessage](https://github.com/dotnet/extensions/blob/main/src/Libraries/Microsoft.Extensions.AI.Abstractions/ChatCompletion/ChatMessage.cs) and [AIContent](https://github.com/dotnet/extensions/blob/main/src/Libraries/Microsoft.Extensions.AI.Abstractions/Contents/AIContent.cs)), durable agent state uses its own, parallel, types in order to (1) better manage the versioning and compatibility of serialized state over time, (2) account for agent implementations across languages/platforms (e.g. .NET and Python), as well as (3) ensure consistency for external tools that make use of state data. + +> When new AI content types are added to the Microsoft Agent Framework, equivalent types should be added to the entity state schema as well. The durable agent state "unknown" type can be used when an AI content type is encountered but no equivalent type exists. + +## State Versioning + +The serialized state contains a root `schemaVersion` property, which represents the version of the schema used to serialize data in that state (represented by the `data` property). + +Some versioning considerations: + +- Versions should use semver notation (e.g. `".."`) +- Durable agents should use the version property to determine how to deserialize that state and should not attempt to deserialize semver-incompatible versions +- Newer versions of durable agents should strive to be compatible with older schema versions (e.g. new properties and objects should be optional) +- Durable agents should preserve existing, but unrecognized, properties when serializing state + +## Sample State + +```json +{ + "schemaVersion": "1.0.0", + "data": { + "conversationHistory": [ + { + "$type": "request", + "responseType": "text", + "correlationId": "c338f064f4b44b8d9c21a66e3cda41b2", + "createdAt": "2025-11-04T19:33:05.245476+00:00", + "messages": [ + { + "contents": [ + { + "$type": "text", + "text": "Start the documentation generation workflow for the product \u0027Goldbrew Coffee\u0027" + } + ], + "role": "user" + } + ] + }, + { + "$type": "response", + "usage": { + "inputTokenCount": 595, + "outputTokenCount": 63, + "totalTokenCount": 658 + }, + "correlationId": "c338f064f4b44b8d9c21a66e3cda41b2", + "createdAt": "2025-11-04T19:33:10.47008+00:00", + "messages": [ + { + "authorName": "OrchestratorAgent", + "createdAt": "2025-11-04T19:33:10+00:00", + "contents": [ + { + "$type": "functionCall", + "arguments": { + "productName": "Goldbrew Coffee" + }, + "callId": "call_qWk9Ay4doKYrUBoADK8MBwHf", + "name": "StartDocumentGeneration" + } + ], + "role": "assistant" + }, + { + "authorName": "OrchestratorAgent", + "createdAt": "2025-11-04T19:33:10.47008+00:00", + "contents": [ + { + "$type": "functionResult", + "callId": "call_qWk9Ay4doKYrUBoADK8MBwHf", + "result": "8b835e8f2a6f40faabdba33bd8fd8c74" + } + ], + "role": "tool" + }, + { + "authorName": "OrchestratorAgent", + "createdAt": "2025-11-04T19:33:10+00:00", + "contents": [ + { + "$type": "text", + "text": "The documentation generation workflow for the product \u0022Goldbrew Coffee\u0022 has been started. You can request updates on its status or provide additional input anytime during the process. Let me know how you\u2019d like to proceed!" + } + ], + "role": "assistant" + } + ] + }, + { + "$type": "request", + "responseType": "text", + "correlationId": "71f35b7add6b403fadd0db8a7c137b58", + "createdAt": "2025-11-04T19:33:11.903413+00:00", + "messages": [ + { + "contents": [ + { + "$type": "text", + "text": "Tell the user that you\u0027re starting to gather information for product \u0027Goldbrew Coffee\u0027." + } + ], + "role": "system" + } + ] + }, + { + "$type": "response", + "usage": { + "inputTokenCount": 396, + "outputTokenCount": 48, + "totalTokenCount": 444 + }, + "correlationId": "71f35b7add6b403fadd0db8a7c137b58", + "createdAt": "2025-11-04T19:33:12+00:00", + "messages": [ + { + "authorName": "OrchestratorAgent", + "createdAt": "2025-11-04T19:33:12+00:00", + "contents": [ + { + "$type": "text", + "text": "I am starting to gather information to create product documentation for \u0027Goldbrew Coffee\u0027. If you have any specific details, key features, or requirements you\u0027d like included, please share them. Otherwise, I\u0027ll continue with the standard documentation process." + } + ], + "role": "assistant" + } + ] + } + ] + } +} +``` + +## State Consumers + +Additional tools may make use of durable agent state. Significant changes to the state schema may need corresponding changes to those applications. + +### Durable Task Scheduler Dashboard + +The [Durable Task Scheduler (DTS)](https://learn.microsoft.com/azure/azure-functions/durable/durable-task-scheduler/durable-task-scheduler) Dashboard, while providing general UX for management of durable orchestrations and entities, also has UX specific to the use of durable agents. diff --git a/dotnet/src/Microsoft.Agents.AI.DurableTask/TaskOrchestrationContextExtensions.cs b/dotnet/src/Microsoft.Agents.AI.DurableTask/TaskOrchestrationContextExtensions.cs new file mode 100644 index 0000000000..63f491cf48 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.DurableTask/TaskOrchestrationContextExtensions.cs @@ -0,0 +1,47 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.ComponentModel; +using Microsoft.DurableTask; + +namespace Microsoft.Agents.AI.DurableTask; + +/// +/// Agent-related extension methods for the class. +/// +[EditorBrowsable(EditorBrowsableState.Never)] +public static class TaskOrchestrationContextExtensions +{ + /// + /// Gets a for interacting with hosted agents within an orchestration. + /// + /// The orchestration context. + /// The name of the agent. + /// Thrown when is null or empty. + /// A that can be used to interact with the agent. + public static DurableAIAgent GetAgent( + this TaskOrchestrationContext context, + string agentName) + { + ArgumentException.ThrowIfNullOrEmpty(agentName); + return new DurableAIAgent(context, agentName); + } + + /// + /// Generates an for an agent. + /// + /// + /// This method is deterministic and safe for use in an orchestration context. + /// + /// The orchestration context. + /// The name of the agent. + /// Thrown when is null or empty. + /// The generated agent session ID. + internal static AgentSessionId NewAgentSessionId( + this TaskOrchestrationContext context, + string agentName) + { + ArgumentException.ThrowIfNullOrEmpty(agentName); + + return new AgentSessionId(agentName, context.NewGuid().ToString("N")); + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting.A2A.AspNetCore/EndpointRouteBuilderExtensions.cs b/dotnet/src/Microsoft.Agents.AI.Hosting.A2A.AspNetCore/EndpointRouteBuilderExtensions.cs index cae9801148..b6b9c4da48 100644 --- a/dotnet/src/Microsoft.Agents.AI.Hosting.A2A.AspNetCore/EndpointRouteBuilderExtensions.cs +++ b/dotnet/src/Microsoft.Agents.AI.Hosting.A2A.AspNetCore/EndpointRouteBuilderExtensions.cs @@ -18,6 +18,21 @@ namespace Microsoft.AspNetCore.Builder; /// public static class MicrosoftAgentAIHostingA2AEndpointRouteBuilderExtensions { + /// + /// Attaches A2A (Agent2Agent) communication capabilities via Message processing to the specified web application. + /// + /// The to add the A2A endpoints to. + /// The configuration builder for . + /// The route group to use for A2A endpoints. + /// Configured for A2A integration. + /// + /// This method can be used to access A2A agents that support the + /// Curated Registries (Catalog-Based Discovery) + /// discovery mechanism. + /// + public static IEndpointConventionBuilder MapA2A(this IEndpointRouteBuilder endpoints, IHostedAgentBuilder agentBuilder, string path) + => endpoints.MapA2A(agentBuilder, path, _ => { }); + /// /// Attaches A2A (Agent2Agent) communication capabilities via Message processing to the specified web application. /// @@ -28,6 +43,25 @@ public static class MicrosoftAgentAIHostingA2AEndpointRouteBuilderExtensions public static IEndpointConventionBuilder MapA2A(this IEndpointRouteBuilder endpoints, string agentName, string path) => endpoints.MapA2A(agentName, path, _ => { }); + /// + /// Attaches A2A (Agent2Agent) communication capabilities via Message processing to the specified web application. + /// + /// The to add the A2A endpoints to. + /// The configuration builder for . + /// The route group to use for A2A endpoints. + /// The callback to configure . + /// Configured for A2A integration. + /// + /// This method can be used to access A2A agents that support the + /// Curated Registries (Catalog-Based Discovery) + /// discovery mechanism. + /// + public static IEndpointConventionBuilder MapA2A(this IEndpointRouteBuilder endpoints, IHostedAgentBuilder agentBuilder, string path, Action configureTaskManager) + { + ArgumentNullException.ThrowIfNull(agentBuilder); + return endpoints.MapA2A(agentBuilder.Name, path, configureTaskManager); + } + /// /// Attaches A2A (Agent2Agent) communication capabilities via Message processing to the specified web application. /// @@ -38,10 +72,27 @@ public static IEndpointConventionBuilder MapA2A(this IEndpointRouteBuilder endpo /// Configured for A2A integration. public static IEndpointConventionBuilder MapA2A(this IEndpointRouteBuilder endpoints, string agentName, string path, Action configureTaskManager) { + ArgumentNullException.ThrowIfNull(endpoints); var agent = endpoints.ServiceProvider.GetRequiredKeyedService(agentName); return endpoints.MapA2A(agent, path, configureTaskManager); } + /// + /// Attaches A2A (Agent2Agent) communication capabilities via Message processing to the specified web application. + /// + /// The to add the A2A endpoints to. + /// The configuration builder for . + /// The route group to use for A2A endpoints. + /// Agent card info to return on query. + /// Configured for A2A integration. + /// + /// This method can be used to access A2A agents that support the + /// Curated Registries (Catalog-Based Discovery) + /// discovery mechanism. + /// + public static IEndpointConventionBuilder MapA2A(this IEndpointRouteBuilder endpoints, IHostedAgentBuilder agentBuilder, string path, AgentCard agentCard) + => endpoints.MapA2A(agentBuilder, path, agentCard, _ => { }); + /// /// Attaches A2A (Agent2Agent) communication capabilities via Message processing to the specified web application. /// @@ -58,6 +109,26 @@ public static IEndpointConventionBuilder MapA2A(this IEndpointRouteBuilder endpo public static IEndpointConventionBuilder MapA2A(this IEndpointRouteBuilder endpoints, string agentName, string path, AgentCard agentCard) => endpoints.MapA2A(agentName, path, agentCard, _ => { }); + /// + /// Attaches A2A (Agent2Agent) communication capabilities via Message processing to the specified web application. + /// + /// The to add the A2A endpoints to. + /// The configuration builder for . + /// The route group to use for A2A endpoints. + /// Agent card info to return on query. + /// The callback to configure . + /// Configured for A2A integration. + /// + /// This method can be used to access A2A agents that support the + /// Curated Registries (Catalog-Based Discovery) + /// discovery mechanism. + /// + public static IEndpointConventionBuilder MapA2A(this IEndpointRouteBuilder endpoints, IHostedAgentBuilder agentBuilder, string path, AgentCard agentCard, Action configureTaskManager) + { + ArgumentNullException.ThrowIfNull(agentBuilder); + return endpoints.MapA2A(agentBuilder.Name, path, agentCard, configureTaskManager); + } + /// /// Attaches A2A (Agent2Agent) communication capabilities via Message processing to the specified web application. /// @@ -74,6 +145,7 @@ public static IEndpointConventionBuilder MapA2A(this IEndpointRouteBuilder endpo /// public static IEndpointConventionBuilder MapA2A(this IEndpointRouteBuilder endpoints, string agentName, string path, AgentCard agentCard, Action configureTaskManager) { + ArgumentNullException.ThrowIfNull(endpoints); var agent = endpoints.ServiceProvider.GetRequiredKeyedService(agentName); return endpoints.MapA2A(agent, path, agentCard, configureTaskManager); } @@ -98,6 +170,9 @@ public static IEndpointConventionBuilder MapA2A(this IEndpointRouteBuilder endpo /// Configured for A2A integration. public static IEndpointConventionBuilder MapA2A(this IEndpointRouteBuilder endpoints, AIAgent agent, string path, Action configureTaskManager) { + ArgumentNullException.ThrowIfNull(endpoints); + ArgumentNullException.ThrowIfNull(agent); + var loggerFactory = endpoints.ServiceProvider.GetRequiredService(); var agentThreadStore = endpoints.ServiceProvider.GetKeyedService(agent.Name); var taskManager = agent.MapA2A(loggerFactory: loggerFactory, agentThreadStore: agentThreadStore); @@ -139,6 +214,9 @@ public static IEndpointConventionBuilder MapA2A(this IEndpointRouteBuilder endpo /// public static IEndpointConventionBuilder MapA2A(this IEndpointRouteBuilder endpoints, AIAgent agent, string path, AgentCard agentCard, Action configureTaskManager) { + ArgumentNullException.ThrowIfNull(endpoints); + ArgumentNullException.ThrowIfNull(agent); + var loggerFactory = endpoints.ServiceProvider.GetRequiredService(); var agentThreadStore = endpoints.ServiceProvider.GetKeyedService(agent.Name); var taskManager = agent.MapA2A(agentCard: agentCard, agentThreadStore: agentThreadStore, loggerFactory: loggerFactory); diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting.A2A.AspNetCore/Microsoft.Agents.AI.Hosting.A2A.AspNetCore.csproj b/dotnet/src/Microsoft.Agents.AI.Hosting.A2A.AspNetCore/Microsoft.Agents.AI.Hosting.A2A.AspNetCore.csproj index c23796ad56..093c5e0cfb 100644 --- a/dotnet/src/Microsoft.Agents.AI.Hosting.A2A.AspNetCore/Microsoft.Agents.AI.Hosting.A2A.AspNetCore.csproj +++ b/dotnet/src/Microsoft.Agents.AI.Hosting.A2A.AspNetCore/Microsoft.Agents.AI.Hosting.A2A.AspNetCore.csproj @@ -1,8 +1,7 @@ - $(ProjectsCoreTargetFrameworks) - $(ProjectsDebugCoreTargetFrameworks) + $(TargetFrameworksCore) Microsoft.Agents.AI.Hosting.A2A.AspNetCore preview @@ -11,11 +10,12 @@ - - - + + + + diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting.A2A/AIAgentExtensions.cs b/dotnet/src/Microsoft.Agents.AI.Hosting.A2A/AIAgentExtensions.cs index 43376d8fb2..c54af66bb8 100644 --- a/dotnet/src/Microsoft.Agents.AI.Hosting.A2A/AIAgentExtensions.cs +++ b/dotnet/src/Microsoft.Agents.AI.Hosting.A2A/AIAgentExtensions.cs @@ -83,7 +83,16 @@ public static ITaskManager MapA2A( { // A2A SDK assigns the url on its own // we can help user if they did not set Url explicitly. - agentCard.Url ??= context; + if (string.IsNullOrEmpty(agentCard.Url)) + { + var agentCardUrl = context.TrimEnd('/'); + if (!context.EndsWith("/v1/card", StringComparison.Ordinal)) + { + agentCardUrl += "/v1/card"; + } + + agentCard.Url = agentCardUrl; + } return Task.FromResult(agentCard); }; diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting.A2A/Microsoft.Agents.AI.Hosting.A2A.csproj b/dotnet/src/Microsoft.Agents.AI.Hosting.A2A/Microsoft.Agents.AI.Hosting.A2A.csproj index 5076e25c05..a0d66cc1d5 100644 --- a/dotnet/src/Microsoft.Agents.AI.Hosting.A2A/Microsoft.Agents.AI.Hosting.A2A.csproj +++ b/dotnet/src/Microsoft.Agents.AI.Hosting.A2A/Microsoft.Agents.AI.Hosting.A2A.csproj @@ -1,8 +1,7 @@ - $(ProjectsCoreTargetFrameworks) - $(ProjectsDebugCoreTargetFrameworks) + $(TargetFrameworksCore) Microsoft.Agents.AI.Hosting.A2A preview Microsoft Agent Framework Hosting A2A @@ -17,9 +16,6 @@ - - - @@ -29,6 +25,6 @@ - + diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore/AGUIChatResponseUpdateStreamExtensions.cs b/dotnet/src/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore/AGUIChatResponseUpdateStreamExtensions.cs new file mode 100644 index 0000000000..c824331f60 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore/AGUIChatResponseUpdateStreamExtensions.cs @@ -0,0 +1,90 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Collections.Generic; +using System.Linq; +using System.Runtime.CompilerServices; +using System.Threading; +using System.Threading.Tasks; +using Microsoft.Extensions.AI; + +namespace Microsoft.Agents.AI.Hosting.AGUI.AspNetCore; + +internal static class AGUIChatResponseUpdateStreamExtensions +{ + public static async IAsyncEnumerable FilterServerToolsFromMixedToolInvocationsAsync( + this IAsyncEnumerable updates, + List? clientTools, + [EnumeratorCancellation] CancellationToken cancellationToken) + { + if (clientTools is null || clientTools.Count == 0) + { + await foreach (var update in updates.WithCancellation(cancellationToken)) + { + yield return update; + } + yield break; + } + + var set = new HashSet(clientTools.Count); + foreach (var tool in clientTools) + { + set.Add(tool.Name); + } + + await foreach (var update in updates.WithCancellation(cancellationToken)) + { + if (update.FinishReason == ChatFinishReason.ToolCalls) + { + var containsClientTools = false; + var containsServerTools = false; + for (var i = update.Contents.Count - 1; i >= 0; i--) + { + var content = update.Contents[i]; + if (content is FunctionCallContent functionCallContent) + { + containsClientTools |= set.Contains(functionCallContent.Name); + containsServerTools |= !set.Contains(functionCallContent.Name); + if (containsClientTools && containsServerTools) + { + break; + } + } + } + + if (containsClientTools && containsServerTools) + { + var newContents = new List(); + for (var i = update.Contents.Count - 1; i >= 0; i--) + { + var content = update.Contents[i]; + if (content is not FunctionCallContent fcc || + set.Contains(fcc.Name)) + { + newContents.Add(content); + } + } + + yield return new ChatResponseUpdate(update.Role, newContents) + { + ConversationId = update.ConversationId, + ResponseId = update.ResponseId, + FinishReason = update.FinishReason, + AdditionalProperties = update.AdditionalProperties, + AuthorName = update.AuthorName, + CreatedAt = update.CreatedAt, + MessageId = update.MessageId, + ModelId = update.ModelId + }; + } + else + { + yield return update; + } + } + else + { + yield return update; + } + } + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore/AGUIEndpointRouteBuilderExtensions.cs b/dotnet/src/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore/AGUIEndpointRouteBuilderExtensions.cs index 63b71620e2..e20d1ab448 100644 --- a/dotnet/src/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore/AGUIEndpointRouteBuilderExtensions.cs +++ b/dotnet/src/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore/AGUIEndpointRouteBuilderExtensions.cs @@ -1,6 +1,8 @@ // Copyright (c) Microsoft. All rights reserved. +using System.Collections.Generic; using System.Diagnostics.CodeAnalysis; +using System.Linq; using System.Threading; using Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.Shared; using Microsoft.AspNetCore.Builder; @@ -10,6 +12,7 @@ using Microsoft.Extensions.AI; using Microsoft.Extensions.DependencyInjection; using Microsoft.Extensions.Logging; +using Microsoft.Extensions.Options; namespace Microsoft.Agents.AI.Hosting.AGUI.AspNetCore; @@ -37,19 +40,44 @@ public static IEndpointConventionBuilder MapAGUI( return Results.BadRequest(); } - var messages = input.Messages.AsChatMessages(); - var agent = aiAgent; + var jsonOptions = context.RequestServices.GetRequiredService>(); + var jsonSerializerOptions = jsonOptions.Value.SerializerOptions; - var events = agent.RunStreamingAsync( + var messages = input.Messages.AsChatMessages(jsonSerializerOptions); + var clientTools = input.Tools?.AsAITools().ToList(); + + // Create run options with AG-UI context in AdditionalProperties + var runOptions = new ChatClientAgentRunOptions + { + ChatOptions = new ChatOptions + { + Tools = clientTools, + AdditionalProperties = new AdditionalPropertiesDictionary + { + ["ag_ui_state"] = input.State, + ["ag_ui_context"] = input.Context?.Select(c => new KeyValuePair(c.Description, c.Value)).ToArray(), + ["ag_ui_forwarded_properties"] = input.ForwardedProperties, + ["ag_ui_thread_id"] = input.ThreadId, + ["ag_ui_run_id"] = input.RunId + } + } + }; + + // Run the agent and convert to AG-UI events + var events = aiAgent.RunStreamingAsync( messages, + options: runOptions, cancellationToken: cancellationToken) + .AsChatResponseUpdatesAsync() + .FilterServerToolsFromMixedToolInvocationsAsync(clientTools, cancellationToken) .AsAGUIEventStreamAsync( input.ThreadId, input.RunId, + jsonSerializerOptions, cancellationToken); - var logger = context.RequestServices.GetRequiredService>(); - return new AGUIServerSentEventsResult(events, logger); + var sseLogger = context.RequestServices.GetRequiredService>(); + return new AGUIServerSentEventsResult(events, sseLogger); }); } } diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore/AGUIJsonSerializerOptions.cs b/dotnet/src/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore/AGUIJsonSerializerOptions.cs new file mode 100644 index 0000000000..822f6f27e7 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore/AGUIJsonSerializerOptions.cs @@ -0,0 +1,24 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json; + +namespace Microsoft.Agents.AI.Hosting.AGUI.AspNetCore; + +/// +/// Extension methods for JSON serialization. +/// +internal static class AGUIJsonSerializerOptions +{ + /// + /// Gets the default JSON serializer options. + /// + public static JsonSerializerOptions Default { get; } = Create(); + + private static JsonSerializerOptions Create() + { + JsonSerializerOptions options = new(AGUIJsonSerializerContext.Default.Options); + options.TypeInfoResolverChain.Add(AgentAbstractionsJsonUtilities.DefaultOptions.TypeInfoResolver!); + options.MakeReadOnly(); + return options; + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore/AGUIServerSentEventsResult.cs b/dotnet/src/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore/AGUIServerSentEventsResult.cs index a0111605eb..95642771ff 100644 --- a/dotnet/src/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore/AGUIServerSentEventsResult.cs +++ b/dotnet/src/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore/AGUIServerSentEventsResult.cs @@ -20,8 +20,6 @@ internal sealed partial class AGUIServerSentEventsResult : IResult, IDisposable private readonly ILogger _logger; private Utf8JsonWriter? _jsonWriter; - public int? StatusCode => StatusCodes.Status200OK; - internal AGUIServerSentEventsResult(IAsyncEnumerable events, ILogger logger) { this._events = events; diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.csproj b/dotnet/src/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.csproj index 522f7f77de..8f6ac4de24 100644 --- a/dotnet/src/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.csproj +++ b/dotnet/src/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.csproj @@ -1,8 +1,7 @@ - $(ProjectsCoreTargetFrameworks) - $(ProjectsDebugCoreTargetFrameworks) + $(TargetFrameworksCore) Microsoft.Agents.AI.Hosting.AGUI.AspNetCore preview $(DefineConstants);ASPNETCORE @@ -12,11 +11,6 @@ - - - false - - Microsoft Agent Framework Hosting AG-UI ASP.NET Core @@ -29,9 +23,6 @@ - - - @@ -39,6 +30,10 @@ + + + + diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore/ServiceCollectionExtensions.cs b/dotnet/src/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore/ServiceCollectionExtensions.cs new file mode 100644 index 0000000000..e159c0727e --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore/ServiceCollectionExtensions.cs @@ -0,0 +1,28 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using Microsoft.Agents.AI; +using Microsoft.Agents.AI.Hosting.AGUI.AspNetCore; +using Microsoft.AspNetCore.Http.Json; + +namespace Microsoft.Extensions.DependencyInjection; + +/// +/// Extension methods for to configure AG-UI support. +/// +public static class MicrosoftAgentAIHostingAGUIServiceCollectionExtensions +{ + /// + /// Adds support for exposing instances via AG-UI. + /// + /// The to configure. + /// The for method chaining. + public static IServiceCollection AddAGUI(this IServiceCollection services) + { + ArgumentNullException.ThrowIfNull(services); + + services.Configure(options => options.SerializerOptions.TypeInfoResolverChain.Add(AGUIJsonSerializerOptions.Default.TypeInfoResolver!)); + + return services; + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/BuiltInFunctionExecutor.cs b/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/BuiltInFunctionExecutor.cs new file mode 100644 index 0000000000..10b1bc54ff --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/BuiltInFunctionExecutor.cs @@ -0,0 +1,107 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Microsoft.Azure.Functions.Worker; +using Microsoft.Azure.Functions.Worker.Context.Features; +using Microsoft.Azure.Functions.Worker.Extensions.Mcp; +using Microsoft.Azure.Functions.Worker.Http; +using Microsoft.Azure.Functions.Worker.Invocation; +using Microsoft.DurableTask.Client; + +namespace Microsoft.Agents.AI.Hosting.AzureFunctions; + +/// +/// This implementation of function executor handles invocations using the built-in static methods for agent HTTP and entity functions. +/// +/// By default, the Azure Functions worker generates function executor and that executor is used for function invocations. +/// But for the dummy HTTP function we create for agents (by augmenting the metadata), that executor will not have the code to handle that function since the entrypoint is a built-in static method. +/// +internal sealed class BuiltInFunctionExecutor : IFunctionExecutor +{ + public async ValueTask ExecuteAsync(FunctionContext context) + { + ArgumentNullException.ThrowIfNull(context); + + // Acquire the input binding feature (fail fast if missing rather than null-forgiving operator). + IFunctionInputBindingFeature? functionInputBindingFeature = context.Features.Get() ?? + throw new InvalidOperationException("Function input binding feature is not available on the current context."); + + FunctionInputBindingResult? inputBindingResults = await functionInputBindingFeature.BindFunctionInputAsync(context); + if (inputBindingResults is not { Values: { } values }) + { + throw new InvalidOperationException($"Function input binding failed for the invocation {context.InvocationId}"); + } + + HttpRequestData? httpRequestData = null; + TaskEntityDispatcher? dispatcher = null; + DurableTaskClient? durableTaskClient = null; + ToolInvocationContext? mcpToolInvocationContext = null; + + foreach (var binding in values) + { + switch (binding) + { + case HttpRequestData request: + httpRequestData = request; + break; + case TaskEntityDispatcher entityDispatcher: + dispatcher = entityDispatcher; + break; + case DurableTaskClient client: + durableTaskClient = client; + break; + case ToolInvocationContext toolContext: + mcpToolInvocationContext = toolContext; + break; + } + } + + if (durableTaskClient is null) + { + // This is not expected to happen since all built-in functions are + // expected to have a Durable Task client binding. + throw new InvalidOperationException($"Durable Task client binding is missing for the invocation {context.InvocationId}."); + } + + if (context.FunctionDefinition.EntryPoint == BuiltInFunctions.RunAgentHttpFunctionEntryPoint) + { + if (httpRequestData == null) + { + throw new InvalidOperationException($"HTTP request data binding is missing for the invocation {context.InvocationId}."); + } + + context.GetInvocationResult().Value = await BuiltInFunctions.RunAgentHttpAsync( + httpRequestData, + durableTaskClient, + context); + return; + } + + if (context.FunctionDefinition.EntryPoint == BuiltInFunctions.RunAgentEntityFunctionEntryPoint) + { + if (dispatcher is null) + { + throw new InvalidOperationException($"Task entity dispatcher binding is missing for the invocation {context.InvocationId}."); + } + + await BuiltInFunctions.InvokeAgentAsync( + dispatcher, + durableTaskClient, + context); + return; + } + + if (context.FunctionDefinition.EntryPoint == BuiltInFunctions.RunAgentMcpToolFunctionEntryPoint) + { + if (mcpToolInvocationContext is null) + { + throw new InvalidOperationException($"MCP tool invocation context binding is missing for the invocation {context.InvocationId}."); + } + + context.GetInvocationResult().Value = + await BuiltInFunctions.RunMcpToolAsync(mcpToolInvocationContext, durableTaskClient, context); + return; + } + + throw new InvalidOperationException($"Unsupported function entry point '{context.FunctionDefinition.EntryPoint}' for invocation {context.InvocationId}."); + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/BuiltInFunctions.cs b/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/BuiltInFunctions.cs new file mode 100644 index 0000000000..ebd378ac3b --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/BuiltInFunctions.cs @@ -0,0 +1,374 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Net; +using System.Text.Json.Serialization; +using Microsoft.Agents.AI.DurableTask; +using Microsoft.Azure.Functions.Worker; +using Microsoft.Azure.Functions.Worker.Extensions.Mcp; +using Microsoft.Azure.Functions.Worker.Http; +using Microsoft.DurableTask.Client; +using Microsoft.Extensions.AI; +using Microsoft.Extensions.DependencyInjection; + +namespace Microsoft.Agents.AI.Hosting.AzureFunctions; + +internal static class BuiltInFunctions +{ + internal const string HttpPrefix = "http-"; + internal const string McpToolPrefix = "mcptool-"; + + internal static readonly string RunAgentHttpFunctionEntryPoint = $"{typeof(BuiltInFunctions).FullName!}.{nameof(RunAgentHttpAsync)}"; + internal static readonly string RunAgentEntityFunctionEntryPoint = $"{typeof(BuiltInFunctions).FullName!}.{nameof(InvokeAgentAsync)}"; + internal static readonly string RunAgentMcpToolFunctionEntryPoint = $"{typeof(BuiltInFunctions).FullName!}.{nameof(RunMcpToolAsync)}"; + + // Exposed as an entity trigger via AgentFunctionsProvider + public static async Task InvokeAgentAsync( + [EntityTrigger] TaskEntityDispatcher dispatcher, + [DurableClient] DurableTaskClient client, + FunctionContext functionContext) + { + // This should never be null except if the function trigger is misconfigured. + ArgumentNullException.ThrowIfNull(dispatcher); + ArgumentNullException.ThrowIfNull(client); + ArgumentNullException.ThrowIfNull(functionContext); + + // Create a combined service provider that includes both the existing services + // and the DurableTaskClient instance + IServiceProvider combinedServiceProvider = new CombinedServiceProvider(functionContext.InstanceServices, client); + + // This method is the entry point for the agent entity. + // It will be invoked by the Azure Functions runtime when the entity is called. + await dispatcher.DispatchAsync(new AgentEntity(combinedServiceProvider, functionContext.CancellationToken)); + } + + public static async Task RunAgentHttpAsync( + [HttpTrigger] HttpRequestData req, + [DurableClient] DurableTaskClient client, + FunctionContext context) + { + // Parse request body - support both JSON and plain text + string? message = null; + string? threadIdFromBody = null; + + if (req.Headers.TryGetValues("Content-Type", out IEnumerable? contentTypeValues) && + contentTypeValues.Any(ct => ct.Contains("application/json", StringComparison.OrdinalIgnoreCase))) + { + // Parse JSON body using POCO record + AgentRunRequest? requestBody = await req.ReadFromJsonAsync(context.CancellationToken); + if (requestBody != null) + { + message = requestBody.Message; + threadIdFromBody = requestBody.ThreadId; + } + } + else + { + // Plain text body + message = await req.ReadAsStringAsync(); + } + + // The thread ID can come from query string or JSON body + string? threadIdFromQuery = req.Query["thread_id"]; + + // Validate that if thread_id is specified in both places, they must match + if (!string.IsNullOrEmpty(threadIdFromQuery) && !string.IsNullOrEmpty(threadIdFromBody) && + !string.Equals(threadIdFromQuery, threadIdFromBody, StringComparison.Ordinal)) + { + return await CreateErrorResponseAsync( + req, + context, + HttpStatusCode.BadRequest, + "thread_id specified in both query string and request body must match."); + } + + string? threadIdValue = threadIdFromBody ?? threadIdFromQuery; + + // The thread_id is treated as a session key (not a full session ID). + // If no session key is provided, use the function invocation ID as the session key + // to help correlate the session with the function invocation. + string agentName = GetAgentName(context); + AgentSessionId sessionId = string.IsNullOrEmpty(threadIdValue) + ? new AgentSessionId(agentName, context.InvocationId) + : new AgentSessionId(agentName, threadIdValue); + + if (string.IsNullOrWhiteSpace(message)) + { + return await CreateErrorResponseAsync( + req, + context, + HttpStatusCode.BadRequest, + "Run request cannot be empty."); + } + + // Check if we should wait for response (default is true) + bool waitForResponse = true; + if (req.Headers.TryGetValues("x-ms-wait-for-response", out IEnumerable? waitForResponseValues)) + { + string? waitForResponseValue = waitForResponseValues.FirstOrDefault(); + if (!string.IsNullOrEmpty(waitForResponseValue) && bool.TryParse(waitForResponseValue, out bool parsedValue)) + { + waitForResponse = parsedValue; + } + } + + AIAgent agentProxy = client.AsDurableAgentProxy(context, agentName); + + DurableAgentRunOptions options = new() { IsFireAndForget = !waitForResponse }; + + if (waitForResponse) + { + AgentRunResponse agentResponse = await agentProxy.RunAsync( + message: new ChatMessage(ChatRole.User, message), + thread: new DurableAgentThread(sessionId), + options: options, + cancellationToken: context.CancellationToken); + + return await CreateSuccessResponseAsync( + req, + context, + HttpStatusCode.OK, + sessionId.Key, + agentResponse); + } + + // Fire and forget - return 202 Accepted + await agentProxy.RunAsync( + message: new ChatMessage(ChatRole.User, message), + thread: new DurableAgentThread(sessionId), + options: options, + cancellationToken: context.CancellationToken); + + return await CreateAcceptedResponseAsync( + req, + context, + sessionId.Key); + } + + public static async Task RunMcpToolAsync( + [McpToolTrigger("BuiltInMcpTool")] ToolInvocationContext context, + [DurableClient] DurableTaskClient client, + FunctionContext functionContext) + { + if (context.Arguments is null) + { + throw new ArgumentException("MCP Tool invocation is missing required arguments."); + } + + if (!context.Arguments.TryGetValue("query", out object? queryObj) || queryObj is not string query) + { + throw new ArgumentException("MCP Tool invocation is missing required 'query' argument of type string."); + } + + string agentName = context.Name; + + // Derive session id: try to parse provided threadId, otherwise create a new one. + AgentSessionId sessionId = context.Arguments.TryGetValue("threadId", out object? threadObj) && threadObj is string threadId && !string.IsNullOrWhiteSpace(threadId) + ? AgentSessionId.Parse(threadId) + : new AgentSessionId(agentName, functionContext.InvocationId); + + AIAgent agentProxy = client.AsDurableAgentProxy(functionContext, agentName); + + AgentRunResponse agentResponse = await agentProxy.RunAsync( + message: new ChatMessage(ChatRole.User, query), + thread: new DurableAgentThread(sessionId), + options: null); + + return agentResponse.Text; + } + + /// + /// Creates an error response with the specified status code and error message. + /// + /// The HTTP request data. + /// The function context. + /// The HTTP status code. + /// The error message. + /// The HTTP response data containing the error. + private static async Task CreateErrorResponseAsync( + HttpRequestData req, + FunctionContext context, + HttpStatusCode statusCode, + string errorMessage) + { + HttpResponseData response = req.CreateResponse(statusCode); + bool acceptsJson = req.Headers.TryGetValues("Accept", out IEnumerable? acceptValues) && + acceptValues.Contains("application/json", StringComparer.OrdinalIgnoreCase); + + if (acceptsJson) + { + ErrorResponse errorResponse = new((int)statusCode, errorMessage); + await response.WriteAsJsonAsync(errorResponse, context.CancellationToken); + } + else + { + response.Headers.Add("Content-Type", "text/plain"); + await response.WriteStringAsync(errorMessage, context.CancellationToken); + } + + return response; + } + + /// + /// Creates a successful agent run response with the agent's response. + /// + /// The HTTP request data. + /// The function context. + /// The HTTP status code (typically 200 OK). + /// The thread ID for the conversation. + /// The agent's response. + /// The HTTP response data containing the success response. + private static async Task CreateSuccessResponseAsync( + HttpRequestData req, + FunctionContext context, + HttpStatusCode statusCode, + string threadId, + AgentRunResponse agentResponse) + { + HttpResponseData response = req.CreateResponse(statusCode); + response.Headers.Add("x-ms-thread-id", threadId); + + bool acceptsJson = req.Headers.TryGetValues("Accept", out IEnumerable? acceptValues) && + acceptValues.Contains("application/json", StringComparer.OrdinalIgnoreCase); + + if (acceptsJson) + { + AgentRunSuccessResponse successResponse = new((int)statusCode, threadId, agentResponse); + await response.WriteAsJsonAsync(successResponse, context.CancellationToken); + } + else + { + response.Headers.Add("Content-Type", "text/plain"); + await response.WriteStringAsync(agentResponse.Text, context.CancellationToken); + } + + return response; + } + + /// + /// Creates an accepted (fire-and-forget) agent run response. + /// + /// The HTTP request data. + /// The function context. + /// The thread ID for the conversation. + /// The HTTP response data containing the accepted response. + private static async Task CreateAcceptedResponseAsync( + HttpRequestData req, + FunctionContext context, + string threadId) + { + HttpResponseData response = req.CreateResponse(HttpStatusCode.Accepted); + response.Headers.Add("x-ms-thread-id", threadId); + + bool acceptsJson = req.Headers.TryGetValues("Accept", out IEnumerable? acceptValues) && + acceptValues.Contains("application/json", StringComparer.OrdinalIgnoreCase); + + if (acceptsJson) + { + AgentRunAcceptedResponse acceptedResponse = new((int)HttpStatusCode.Accepted, threadId); + await response.WriteAsJsonAsync(acceptedResponse, context.CancellationToken); + } + else + { + response.Headers.Add("Content-Type", "text/plain"); + await response.WriteStringAsync("Request accepted.", context.CancellationToken); + } + + return response; + } + + private static string GetAgentName(FunctionContext context) + { + // Check if the function name starts with the HttpPrefix + string functionName = context.FunctionDefinition.Name; + if (!functionName.StartsWith(HttpPrefix, StringComparison.Ordinal)) + { + // This should never happen because the function metadata provider ensures + // that the function name starts with the HttpPrefix (http-). + throw new InvalidOperationException( + $"Built-in HTTP trigger function name '{functionName}' does not start with '{HttpPrefix}'."); + } + + // Remove the HttpPrefix from the function name to get the agent name. + return functionName[HttpPrefix.Length..]; + } + + /// + /// Represents a request to run an agent. + /// + /// The message to send to the agent. + /// The optional thread ID to continue a conversation. + private sealed record AgentRunRequest( + [property: JsonPropertyName("message")] string? Message, + [property: JsonPropertyName("thread_id")] string? ThreadId); + + /// + /// Represents an error response. + /// + /// The HTTP status code. + /// The error message. + private sealed record ErrorResponse( + [property: JsonPropertyName("status")] int Status, + [property: JsonPropertyName("error")] string Error); + + /// + /// Represents a successful agent run response. + /// + /// The HTTP status code. + /// The thread ID for the conversation. + /// The agent response. + private sealed record AgentRunSuccessResponse( + [property: JsonPropertyName("status")] int Status, + [property: JsonPropertyName("thread_id")] string ThreadId, + [property: JsonPropertyName("response")] AgentRunResponse Response); + + /// + /// Represents an accepted (fire-and-forget) agent run response. + /// + /// The HTTP status code. + /// The thread ID for the conversation. + private sealed record AgentRunAcceptedResponse( + [property: JsonPropertyName("status")] int Status, + [property: JsonPropertyName("thread_id")] string ThreadId); + + /// + /// A service provider that combines the original service provider with an additional DurableTaskClient instance. + /// + private sealed class CombinedServiceProvider(IServiceProvider originalProvider, DurableTaskClient client) + : IServiceProvider, IKeyedServiceProvider + { + private readonly IServiceProvider _originalProvider = originalProvider; + private readonly DurableTaskClient _client = client; + + public object? GetKeyedService(Type serviceType, object? serviceKey) + { + if (this._originalProvider is IKeyedServiceProvider keyedProvider) + { + return keyedProvider.GetKeyedService(serviceType, serviceKey); + } + + return null; + } + + public object GetRequiredKeyedService(Type serviceType, object? serviceKey) + { + if (this._originalProvider is IKeyedServiceProvider keyedProvider) + { + return keyedProvider.GetRequiredKeyedService(serviceType, serviceKey); + } + + throw new InvalidOperationException("The original service provider does not support keyed services."); + } + + public object? GetService(Type serviceType) + { + // If the requested service is DurableTaskClient, return our instance + if (serviceType == typeof(DurableTaskClient)) + { + return this._client; + } + + // Otherwise try to get the service from the original provider + return this._originalProvider.GetService(serviceType); + } + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/CHANGELOG.md b/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/CHANGELOG.md new file mode 100644 index 0000000000..d32f4bb0e2 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/CHANGELOG.md @@ -0,0 +1,14 @@ +# Release History + +## v1.0.0-preview.251125.1 + +- Added support for .NET 10 ([#2128](https://github.com/microsoft/agent-framework/pull/2128)) +- [BREAKING] Changed `thread_id` in HTTP APIs from entity ID to GUID ([#2260](https://github.com/microsoft/agent-framework/pull/2260)) + +## v1.0.0-preview.251114.1 + +- Added friendly error message when running durable agent that isn't registered ([#2214](https://github.com/microsoft/agent-framework/pull/2214)) + +## v1.0.0-preview.251112.1 + +- Initial public release ([#1916](https://github.com/microsoft/agent-framework/pull/1916)) diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/DefaultFunctionsAgentOptionsProvider.cs b/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/DefaultFunctionsAgentOptionsProvider.cs new file mode 100644 index 0000000000..1039fb5aec --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/DefaultFunctionsAgentOptionsProvider.cs @@ -0,0 +1,45 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Diagnostics.CodeAnalysis; + +namespace Microsoft.Agents.AI.Hosting.AzureFunctions; + +/// +/// Provides access to agent-specific options for functions agents by name. +/// Returns default options (HTTP trigger enabled, MCP tool disabled) when no explicit options were configured. +/// +internal sealed class DefaultFunctionsAgentOptionsProvider(IReadOnlyDictionary functionsAgentOptions) + : IFunctionsAgentOptionsProvider +{ + private readonly IReadOnlyDictionary _functionsAgentOptions = + functionsAgentOptions ?? throw new ArgumentNullException(nameof(functionsAgentOptions)); + + // Default options. HTTP trigger enabled, MCP tool disabled. + private static readonly FunctionsAgentOptions s_defaultOptions = new() + { + HttpTrigger = { IsEnabled = true }, + McpToolTrigger = { IsEnabled = false } + }; + + /// + /// Attempts to retrieve the options associated with the specified agent name. + /// If not found, a default options instance (with HTTP trigger enabled) is returned. + /// + /// The name of the agent whose options are to be retrieved. Cannot be null or empty. + /// The options for the specified agent. Will never be null. + /// Always true. Returns configured options if present; otherwise default fallback options. + public bool TryGet(string agentName, [NotNullWhen(true)] out FunctionsAgentOptions? options) + { + ArgumentException.ThrowIfNullOrEmpty(agentName); + + if (this._functionsAgentOptions.TryGetValue(agentName, out FunctionsAgentOptions? existing)) + { + options = existing; + return true; + } + + // If not defined, return default options. + options = s_defaultOptions; + return true; + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/DurableAgentFunctionMetadataTransformer.cs b/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/DurableAgentFunctionMetadataTransformer.cs new file mode 100644 index 0000000000..cce8fbd1b0 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/DurableAgentFunctionMetadataTransformer.cs @@ -0,0 +1,118 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Microsoft.Agents.AI.DurableTask; +using Microsoft.Azure.Functions.Worker.Core.FunctionMetadata; +using Microsoft.Extensions.Logging; + +namespace Microsoft.Agents.AI.Hosting.AzureFunctions; + +/// +/// Transforms function metadata by registering durable agent functions for each configured agent. +/// +/// This transformer adds both entity trigger and HTTP trigger functions for every agent registered in the application. +internal sealed class DurableAgentFunctionMetadataTransformer : IFunctionMetadataTransformer +{ + private readonly ILogger _logger; + private readonly IReadOnlyDictionary> _agents; + private readonly IServiceProvider _serviceProvider; + private readonly IFunctionsAgentOptionsProvider _functionsAgentOptionsProvider; + +#pragma warning disable IL3000 // Avoid accessing Assembly file path when publishing as a single file - Azure Functions does not use single-file publishing + private static readonly string s_builtInFunctionsScriptFile = Path.GetFileName(typeof(BuiltInFunctions).Assembly.Location); +#pragma warning restore IL3000 + + public DurableAgentFunctionMetadataTransformer( + IReadOnlyDictionary> agents, + ILogger logger, + IServiceProvider serviceProvider, + IFunctionsAgentOptionsProvider functionsAgentOptionsProvider) + { + this._agents = agents ?? throw new ArgumentNullException(nameof(agents)); + this._logger = logger ?? throw new ArgumentNullException(nameof(logger)); + this._serviceProvider = serviceProvider ?? throw new ArgumentNullException(nameof(serviceProvider)); + this._functionsAgentOptionsProvider = functionsAgentOptionsProvider ?? throw new ArgumentNullException(nameof(functionsAgentOptionsProvider)); + } + + public string Name => nameof(DurableAgentFunctionMetadataTransformer); + + public void Transform(IList original) + { + this._logger.LogTransformingFunctionMetadata(original.Count); + + foreach (KeyValuePair> kvp in this._agents) + { + string agentName = kvp.Key; + + this._logger.LogRegisteringTriggerForAgent(agentName, "entity"); + + original.Add(CreateAgentTrigger(agentName)); + + if (this._functionsAgentOptionsProvider.TryGet(agentName, out FunctionsAgentOptions? agentTriggerOptions)) + { + if (agentTriggerOptions.HttpTrigger.IsEnabled) + { + this._logger.LogRegisteringTriggerForAgent(agentName, "http"); + original.Add(CreateHttpTrigger(agentName, $"agents/{agentName}/run")); + } + + if (agentTriggerOptions.McpToolTrigger.IsEnabled) + { + AIAgent agent = kvp.Value(this._serviceProvider); + this._logger.LogRegisteringTriggerForAgent(agentName, "mcpTool"); + original.Add(CreateMcpToolTrigger(agentName, agent.Description)); + } + } + } + } + + private static DefaultFunctionMetadata CreateAgentTrigger(string name) + { + return new DefaultFunctionMetadata() + { + Name = AgentSessionId.ToEntityName(name), + Language = "dotnet-isolated", + RawBindings = + [ + """{"name":"dispatcher","type":"entityTrigger","direction":"In"}""", + """{"name":"client","type":"durableClient","direction":"In"}""" + ], + EntryPoint = BuiltInFunctions.RunAgentEntityFunctionEntryPoint, + ScriptFile = s_builtInFunctionsScriptFile, + }; + } + + private static DefaultFunctionMetadata CreateHttpTrigger(string name, string route) + { + return new DefaultFunctionMetadata() + { + Name = $"{BuiltInFunctions.HttpPrefix}{name}", + Language = "dotnet-isolated", + RawBindings = + [ + $"{{\"name\":\"req\",\"type\":\"httpTrigger\",\"direction\":\"In\",\"authLevel\":\"function\",\"methods\": [\"post\"],\"route\":\"{route}\"}}", + "{\"name\":\"$return\",\"type\":\"http\",\"direction\":\"Out\"}", + "{\"name\":\"client\",\"type\":\"durableClient\",\"direction\":\"In\"}" + ], + EntryPoint = BuiltInFunctions.RunAgentHttpFunctionEntryPoint, + ScriptFile = s_builtInFunctionsScriptFile, + }; + } + + private static DefaultFunctionMetadata CreateMcpToolTrigger(string agentName, string? description) + { + return new DefaultFunctionMetadata + { + Name = $"{BuiltInFunctions.McpToolPrefix}{agentName}", + Language = "dotnet-isolated", + RawBindings = + [ + $$"""{"name":"context","type":"mcpToolTrigger","direction":"In","toolName":"{{agentName}}","description":"{{description}}","toolProperties":"[{\"propertyName\":\"query\",\"propertyType\":\"string\",\"description\":\"The query to send to the agent.\",\"isRequired\":true,\"isArray\":false},{\"propertyName\":\"threadId\",\"propertyType\":\"string\",\"description\":\"Optional thread identifier.\",\"isRequired\":false,\"isArray\":false}]"}""", + """{"name":"query","type":"mcpToolProperty","direction":"In","propertyName":"query","description":"The query to send to the agent","isRequired":true,"dataType":"String","propertyType":"string"}""", + """{"name":"threadId","type":"mcpToolProperty","direction":"In","propertyName":"threadId","description":"The thread identifier.","isRequired":false,"dataType":"String","propertyType":"string"}""", + """{"name":"client","type":"durableClient","direction":"In"}""" + ], + EntryPoint = BuiltInFunctions.RunAgentMcpToolFunctionEntryPoint, + ScriptFile = s_builtInFunctionsScriptFile, + }; + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/DurableAgentsOptionsExtensions.cs b/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/DurableAgentsOptionsExtensions.cs new file mode 100644 index 0000000000..ad21d8f4e1 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/DurableAgentsOptionsExtensions.cs @@ -0,0 +1,137 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Microsoft.Agents.AI.DurableTask; + +namespace Microsoft.Agents.AI.Hosting.AzureFunctions; + +/// +/// Provides extension methods for registering and configuring AI agents in the context of the Azure Functions hosting environment. +/// +public static class DurableAgentsOptionsExtensions +{ + // Registry of agent options. + private static readonly Dictionary s_agentOptions = new(StringComparer.OrdinalIgnoreCase); + + /// + /// Adds an AI agent to the specified DurableAgentsOptions instance and optionally configures agent-specific + /// options. + /// + /// The DurableAgentsOptions instance to which the AI agent will be added. + /// The AI agent to add. The agent's Name property must not be null or empty. + /// An optional delegate to configure agent-specific options. If null, default options are used. + /// The updated instance containing the added AI agent. + public static DurableAgentsOptions AddAIAgent( + this DurableAgentsOptions options, + AIAgent agent, + Action? configure) + { + ArgumentNullException.ThrowIfNull(options); + ArgumentNullException.ThrowIfNull(agent); + ArgumentException.ThrowIfNullOrEmpty(agent.Name); + + // Initialize with default behavior (HTTP trigger enabled) + FunctionsAgentOptions agentOptions = new() { HttpTrigger = { IsEnabled = true } }; + configure?.Invoke(agentOptions); + options.AddAIAgent(agent); + s_agentOptions[agent.Name] = agentOptions; + return options; + } + + /// + /// Adds an AI agent to the specified options and configures trigger support for HTTP and MCP tool invocations. + /// + /// If an agent with the same name already exists in the options, its configuration will be + /// updated. Both triggers can be enabled independently. This method supports method chaining by returning the + /// provided options instance. + /// The options collection to which the AI agent will be added. Cannot be null. + /// The AI agent to add. The agent's Name property must not be null or empty. + /// true to enable an HTTP trigger for the agent; otherwise, false. + /// true to enable an MCP tool trigger for the agent; otherwise, false. + /// The updated instance with the specified AI agent and trigger configuration applied. + public static DurableAgentsOptions AddAIAgent( + this DurableAgentsOptions options, + AIAgent agent, + bool enableHttpTrigger, + bool enableMcpToolTrigger) + { + ArgumentNullException.ThrowIfNull(options); + ArgumentNullException.ThrowIfNull(agent); + ArgumentException.ThrowIfNullOrEmpty(agent.Name); + + FunctionsAgentOptions agentOptions = new(); + agentOptions.HttpTrigger.IsEnabled = enableHttpTrigger; + agentOptions.McpToolTrigger.IsEnabled = enableMcpToolTrigger; + + options.AddAIAgent(agent); + s_agentOptions[agent.Name] = agentOptions; + return options; + } + + /// + /// Registers an AI agent factory with the specified name and optional configuration in the provided + /// DurableAgentsOptions instance. + /// + /// If an agent factory with the same name already exists, its configuration will be replaced. + /// This method enables custom agent registration and configuration for use in durable agent scenarios. + /// The DurableAgentsOptions instance to which the AI agent factory will be added. Cannot be null. + /// The unique name used to identify the AI agent factory. Cannot be null. + /// A delegate that creates an AIAgent instance using the provided IServiceProvider. Cannot be null. + /// An optional action to configure FunctionsAgentOptions for the agent factory. If null, default options are used. + /// The updated DurableAgentsOptions instance containing the registered AI agent factory. + public static DurableAgentsOptions AddAIAgentFactory( + this DurableAgentsOptions options, + string name, + Func factory, + Action? configure) + { + ArgumentNullException.ThrowIfNull(options); + ArgumentNullException.ThrowIfNull(name); + ArgumentNullException.ThrowIfNull(factory); + + // Initialize with default behavior (HTTP trigger enabled) + FunctionsAgentOptions agentOptions = new() { HttpTrigger = { IsEnabled = true } }; + configure?.Invoke(agentOptions); + options.AddAIAgentFactory(name, factory); + s_agentOptions[name] = agentOptions; + return options; + } + + /// + /// Registers an AI agent factory with the specified name and configures trigger options for the agent. + /// + /// If both triggers are disabled, the agent will not be accessible via HTTP or MCP tool + /// endpoints. This method can be used to register multiple agent factories with different configurations. + /// The options object to which the AI agent factory will be added. Cannot be null. + /// The unique name used to identify the AI agent factory. Cannot be null. + /// A delegate that creates an instance of the AI agent using the provided service provider. Cannot be null. + /// true to enable the HTTP trigger for the agent; otherwise, false. + /// true to enable the MCP tool trigger for the agent; otherwise, false. + /// The same DurableAgentsOptions instance, allowing for method chaining. + public static DurableAgentsOptions AddAIAgentFactory( + this DurableAgentsOptions options, + string name, + Func factory, + bool enableHttpTrigger, + bool enableMcpToolTrigger) + { + ArgumentNullException.ThrowIfNull(options); + ArgumentNullException.ThrowIfNull(name); + ArgumentNullException.ThrowIfNull(factory); + + FunctionsAgentOptions agentOptions = new(); + agentOptions.HttpTrigger.IsEnabled = enableHttpTrigger; + agentOptions.McpToolTrigger.IsEnabled = enableMcpToolTrigger; + + options.AddAIAgentFactory(name, factory); + s_agentOptions[name] = agentOptions; + return options; + } + + /// + /// Builds the agentOptions used for dependency injection (read-only copy). + /// + internal static IReadOnlyDictionary GetAgentOptionsSnapshot() + { + return new Dictionary(s_agentOptions, StringComparer.OrdinalIgnoreCase); + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/DurableTaskClientExtensions.cs b/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/DurableTaskClientExtensions.cs new file mode 100644 index 0000000000..0977d756cb --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/DurableTaskClientExtensions.cs @@ -0,0 +1,48 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Microsoft.Agents.AI.DurableTask; +using Microsoft.Azure.Functions.Worker; +using Microsoft.DurableTask.Client; +using Microsoft.Extensions.DependencyInjection; + +namespace Microsoft.Agents.AI.Hosting.AzureFunctions; + +/// +/// Extension methods for the class. +/// +public static class DurableTaskClientExtensions +{ + /// + /// Converts a to a durable agent proxy. + /// + /// The to convert. + /// The for the current function invocation. + /// The name of the agent. + /// A durable agent proxy. + /// Thrown when or is null. + /// Thrown when is null or empty. + /// + /// Thrown when durable agents have not been configured on the service collection. + /// + /// + /// Thrown when the agent has not been registered. + /// + public static AIAgent AsDurableAgentProxy( + this DurableTaskClient durableClient, + FunctionContext context, + string agentName) + { + ArgumentNullException.ThrowIfNull(durableClient); + ArgumentNullException.ThrowIfNull(context); + ArgumentException.ThrowIfNullOrEmpty(agentName); + + // Validate that the agent is registered + DurableTask.ServiceCollectionExtensions.ValidateAgentIsRegistered(context.InstanceServices, agentName); + + DefaultDurableAgentClient agentClient = ActivatorUtilities.CreateInstance( + context.InstanceServices, + durableClient); + + return new DurableAIAgentProxy(agentName, agentClient); + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/FunctionsAgentOptions.cs b/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/FunctionsAgentOptions.cs new file mode 100644 index 0000000000..6ead7d8be5 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/FunctionsAgentOptions.cs @@ -0,0 +1,19 @@ +// Copyright (c) Microsoft. All rights reserved. + +namespace Microsoft.Agents.AI.Hosting.AzureFunctions; + +/// +/// Provides configuration options for enabling and customizing function triggers for an agent. +/// +public sealed class FunctionsAgentOptions +{ + /// + /// Gets or sets the configuration options for the HTTP trigger endpoint. + /// + public HttpTriggerOptions HttpTrigger { get; set; } = new(false); + + /// + /// Gets or sets the options used to configure the MCP tool trigger behavior. + /// + public McpToolTriggerOptions McpToolTrigger { get; set; } = new(false); +} diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/FunctionsApplicationBuilderExtensions.cs b/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/FunctionsApplicationBuilderExtensions.cs new file mode 100644 index 0000000000..e13c6008ea --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/FunctionsApplicationBuilderExtensions.cs @@ -0,0 +1,46 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Microsoft.Agents.AI.DurableTask; +using Microsoft.Azure.Functions.Worker.Builder; +using Microsoft.Azure.Functions.Worker.Core.FunctionMetadata; +using Microsoft.Extensions.DependencyInjection; +using Microsoft.Extensions.DependencyInjection.Extensions; +using Microsoft.Extensions.Hosting; + +namespace Microsoft.Agents.AI.Hosting.AzureFunctions; + +/// +/// Extension methods for the class. +/// +public static class FunctionsApplicationBuilderExtensions +{ + /// + /// Configures the application to use durable agents with a builder pattern. + /// + /// The functions application builder. + /// A delegate to configure the durable agents. + /// The functions application builder. + public static FunctionsApplicationBuilder ConfigureDurableAgents( + this FunctionsApplicationBuilder builder, + Action configure) + { + ArgumentNullException.ThrowIfNull(configure); + + // The main agent services registration is done in Microsoft.DurableTask.Agents. + builder.Services.ConfigureDurableAgents(configure); + + builder.Services.TryAddSingleton(_ => + new DefaultFunctionsAgentOptionsProvider(DurableAgentsOptionsExtensions.GetAgentOptionsSnapshot())); + + builder.Services.AddSingleton(); + + // Handling of built-in function execution for Agent HTTP, MCP tool, or Entity invocations. + builder.UseWhen(static context => + string.Equals(context.FunctionDefinition.EntryPoint, BuiltInFunctions.RunAgentHttpFunctionEntryPoint, StringComparison.Ordinal) || + string.Equals(context.FunctionDefinition.EntryPoint, BuiltInFunctions.RunAgentMcpToolFunctionEntryPoint, StringComparison.Ordinal) || + string.Equals(context.FunctionDefinition.EntryPoint, BuiltInFunctions.RunAgentEntityFunctionEntryPoint, StringComparison.Ordinal)); + builder.Services.AddSingleton(); + + return builder; + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/HttpTriggerOptions.cs b/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/HttpTriggerOptions.cs new file mode 100644 index 0000000000..2a750c3ae5 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/HttpTriggerOptions.cs @@ -0,0 +1,18 @@ +// Copyright (c) Microsoft. All rights reserved. + +namespace Microsoft.Agents.AI.Hosting.AzureFunctions; + +/// +/// Represents configuration options for the HTTP trigger for an agent. +/// +/// +/// Initializes a new instance of the class. +/// +/// Indicates whether the HTTP trigger is enabled for the agent. +public sealed class HttpTriggerOptions(bool isEnabled) +{ + /// + /// Gets or sets a value indicating whether the HTTP trigger is enabled for the agent. + /// + public bool IsEnabled { get; set; } = isEnabled; +} diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/IFunctionsAgentOptionsProvider.cs b/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/IFunctionsAgentOptionsProvider.cs new file mode 100644 index 0000000000..347b4242a3 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/IFunctionsAgentOptionsProvider.cs @@ -0,0 +1,19 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Diagnostics.CodeAnalysis; + +namespace Microsoft.Agents.AI.Hosting.AzureFunctions; + +/// +/// Provides access to function trigger options for agents in the Azure Functions hosting environment. +/// +internal interface IFunctionsAgentOptionsProvider +{ + /// + /// Attempts to get trigger options for the specified agent. + /// + /// The agent name. + /// The resulting options if found. + /// True if options exist; otherwise false. + bool TryGet(string agentName, [NotNullWhen(true)] out FunctionsAgentOptions? options); +} diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/Logs.cs b/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/Logs.cs new file mode 100644 index 0000000000..c49d2b39df --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/Logs.cs @@ -0,0 +1,20 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Microsoft.Extensions.Logging; + +namespace Microsoft.Agents.AI.Hosting.AzureFunctions; + +internal static partial class Logs +{ + [LoggerMessage( + EventId = 100, + Level = LogLevel.Information, + Message = "Transforming function metadata to add durable agent functions. Initial function count: {FunctionCount}")] + public static partial void LogTransformingFunctionMetadata(this ILogger logger, int functionCount); + + [LoggerMessage( + EventId = 101, + Level = LogLevel.Information, + Message = "Registering {TriggerType} function for agent '{AgentName}'")] + public static partial void LogRegisteringTriggerForAgent(this ILogger logger, string agentName, string triggerType); +} diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/McpToolTriggerOptions.cs b/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/McpToolTriggerOptions.cs new file mode 100644 index 0000000000..8e729f6840 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/McpToolTriggerOptions.cs @@ -0,0 +1,18 @@ +// Copyright (c) Microsoft. All rights reserved. + +namespace Microsoft.Agents.AI.Hosting.AzureFunctions; + +/// +/// This class provides configuration options for the MCP tool trigger for an agent. +/// +/// +/// A value indicating whether the MCP tool trigger is enabled for the agent. +/// Set to to enable the trigger; otherwise, . +/// +public sealed class McpToolTriggerOptions(bool isEnabled) +{ + /// + /// Gets or sets a value indicating whether MCP tool trigger is enabled for the agent. + /// + public bool IsEnabled { get; set; } = isEnabled; +} diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/Microsoft.Agents.AI.Hosting.AzureFunctions.csproj b/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/Microsoft.Agents.AI.Hosting.AzureFunctions.csproj new file mode 100644 index 0000000000..ce67c9621e --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/Microsoft.Agents.AI.Hosting.AzureFunctions.csproj @@ -0,0 +1,58 @@ + + + + $(TargetFrameworksCore) + enable + + $(NoWarn);CA2007 + + + + + + + Azure Functions extensions for Microsoft Agent Framework + Provides durable agent hosting and orchestration support for Microsoft Agent Framework workloads. + README.md + + + + + + + + + + + + + + + + + + + + + + + + + + + + + <_Parameter1>Microsoft.Azure.Functions.Extensions.Mcp + <_Parameter2>1.0.0 + + <_Parameter3>true + <_Parameter3_IsLiteral>true + + + diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/Middlewares/BuiltInFunctionExecutionMiddleware.cs b/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/Middlewares/BuiltInFunctionExecutionMiddleware.cs new file mode 100644 index 0000000000..3dc1a58943 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/Middlewares/BuiltInFunctionExecutionMiddleware.cs @@ -0,0 +1,24 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Microsoft.Azure.Functions.Worker; +using Microsoft.Azure.Functions.Worker.Invocation; +using Microsoft.Azure.Functions.Worker.Middleware; + +namespace Microsoft.Agents.AI.Hosting.AzureFunctions; + +/// +/// This middleware sets a custom function executor for invocation of functions that have the built-in method as the entrypoint. +/// +internal sealed class BuiltInFunctionExecutionMiddleware(BuiltInFunctionExecutor builtInFunctionExecutor) + : IFunctionsWorkerMiddleware +{ + private readonly BuiltInFunctionExecutor _builtInFunctionExecutor = builtInFunctionExecutor; + + public async Task Invoke(FunctionContext context, FunctionExecutionDelegate next) + { + // We set our custom function executor for this invocation. + context.Features.Set(this._builtInFunctionExecutor); + + await next(context); + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/README.md b/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/README.md new file mode 100644 index 0000000000..4e819e5985 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Hosting.AzureFunctions/README.md @@ -0,0 +1,177 @@ +# Microsoft.Agents.AI.Hosting.AzureFunctions + +This package adds Azure Functions integration and serverless hosting for Microsoft Agent Framework on Azure Functions. It builds upon the `Microsoft.Agents.AI.DurableTask` package to provide the following capabilities: + +- Stateful, durable execution of agents in distributed, serverless environments +- Automatic conversation history management in supported [Durable Functions backends](https://learn.microsoft.com/azure/azure-functions/durable/durable-functions-storage-providers) +- Long-running agent workflows as "durable orchestrator" functions +- Tools and [dashboards](https://learn.microsoft.com/azure/azure-functions/durable/durable-task-scheduler/durable-task-scheduler-dashboard) for managing and monitoring agents and agent workflows + +## Install the package + +From the command-line: + +```bash +dotnet add package Microsoft.Agents.AI.Hosting.AzureFunctions +``` + +Or directly in your project file: + +```xml + + + +``` + +## Usage Examples + +For a comprehensive tour of all the functionality, concepts, and APIs, check out the [Azure Functions samples](https://github.com/microsoft/agent-framework/tree/main/dotnet/samples/) in the [Microsoft Agent Framework GitHub repository](https://github.com/microsoft/agent-framework). + +### Hosting single agents + +This package provides a `ConfigureDurableAgents` extension method on the `FunctionsApplicationBuilder` class to configure the application to host Microsoft Agent Framework agents. These hosted agents are automatically registered as durable entities with the Durable Task runtime and can be invoked via HTTP or Durable Task orchestrator functions. + +```csharp +// Create agents using the standard Microsoft Agent Framework. +// Invocable via HTTP via http://localhost:7071/api/agents/SpamDetectionAgent/run +AIAgent spamDetector = new AzureOpenAIClient(new Uri(endpoint), new AzureCliCredential()) + .GetChatClient(deploymentName) + .CreateAIAgent( + instructions: "You are a spam detection assistant that identifies spam emails.", + name: "SpamDetectionAgent"); + +AIAgent emailAssistant = new AzureOpenAIClient(new Uri(endpoint), new AzureCliCredential()) + .GetChatClient(deploymentName) + .CreateAIAgent( + instructions: "You are an email assistant that helps users draft responses to emails with professionalism.", + name: "EmailAssistantAgent"); + +// Configure the Functions application to host the agents. +using IHost app = FunctionsApplication + .CreateBuilder(args) + .ConfigureFunctionsWebApplication() + .ConfigureDurableAgents(options => + { + options.AddAIAgent(spamDetector); + options.AddAIAgent(emailAssistant); + }) + .Build(); +app.Run(); +``` + +By default, each agent can be invoked via a built-in HTTP trigger function at the route `http[s]://[host]/api/agents/{agentName}/run`. + +### Orchestrating hosted agents + +This package also provides a set of extension methods such as `GetAgent` on the [`TaskOrchestrationContext`](https://learn.microsoft.com/dotnet/api/microsoft.durabletask.taskorchestrationcontext) class for interacting with hosted agents within orchestrations. + +```csharp +[Function(nameof(SpamDetectionOrchestration))] +public static async Task SpamDetectionOrchestration( + [OrchestrationTrigger] TaskOrchestrationContext context) +{ + Email email = context.GetInput() ?? throw new InvalidOperationException("Email is required"); + + // Get the spam detection agent + DurableAIAgent spamDetectionAgent = context.GetAgent("SpamDetectionAgent"); + AgentThread spamThread = spamDetectionAgent.GetNewThread(); + + // Step 1: Check if the email is spam + AgentRunResponse spamDetectionResponse = await spamDetectionAgent.RunAsync( + message: + $""" + Analyze this email for spam content and return a JSON response with 'is_spam' (boolean) and 'reason' (string) fields: + Email ID: {email.EmailId} + Content: {email.EmailContent} + """, + thread: spamThread); + DetectionResult result = spamDetectionResponse.Result; + + // Step 2: Conditional logic based on spam detection result + if (result.IsSpam) + { + // Handle spam email + return await context.CallActivityAsync(nameof(HandleSpamEmail), result.Reason); + } + else + { + // Generate and send response for legitimate email + DurableAIAgent emailAssistantAgent = context.GetAgent("EmailAssistantAgent"); + AgentThread emailThread = emailAssistantAgent.GetNewThread(); + + AgentRunResponse emailAssistantResponse = await emailAssistantAgent.RunAsync( + message: + $""" + Draft a professional response to this email. Return a JSON response with a 'response' field containing the reply: + + Email ID: {email.EmailId} + Content: {email.EmailContent} + """, + thread: emailThread); + + EmailResponse emailResponse = emailAssistantResponse.Result; + return await context.CallActivityAsync(nameof(SendEmail), emailResponse.Response); + } +} +``` + +### Scheduling orchestrations from custom code tools + +Agents can also schedule and interact with orchestrations from custom code tools. This is useful for long-running tool use cases where orchestrations need to be executed in the context of the agent. + +The `DurableAgentContext.Current` *AsyncLocal* property provides access to the current agent context, which can be used to schedule and interact with orchestrations. + +```csharp +class Tools +{ + [Description("Starts a content generation workflow and returns the instance ID for tracking.")] + public string StartContentGenerationWorkflow( + [Description("The topic for content generation")] string topic) + { + // ContentGenerationWorkflow is an orchestrator function defined in the same project. + string instanceId = DurableAgentContext.Current.ScheduleNewOrchestration( + name: nameof(ContentGenerationWorkflow), + input: topic); + + // Return the instance ID so that it gets added to the LLM context. + return instanceId; + } + + [Description("Gets the status of a content generation workflow.")] + public async Task GetContentGenerationStatus( + [Description("The instance ID of the workflow to check")] string instanceId, + [Description("Whether to include detailed information")] bool includeDetails = true) + { + OrchestrationMetadata? status = await DurableAgentContext.Current.Client.GetOrchestrationStatusAsync( + instanceId, + includeDetails); + return status ?? throw new InvalidOperationException($"Workflow instance '{instanceId}' not found."); + } +} +``` + +These tools are registered with the agent using the `tools` parameter when creating the agent. + +```csharp +Tools tools = new(); +AIAgent agent = new AzureOpenAIClient(new Uri(endpoint), new AzureCliCredential()) + .GetChatClient(deploymentName) + .CreateAIAgent( + instructions: "You are a content generation assistant that helps users generate content.", + name: "ContentGenerationAgent", + tools: [ + AIFunctionFactory.Create(tools.StartContentGenerationWorkflow), + AIFunctionFactory.Create(tools.GetContentGenerationStatus) + ]); + +using IHost app = FunctionsApplication + .CreateBuilder(args) + .ConfigureFunctionsWebApplication() + .ConfigureDurableAgents(options => options.AddAIAgent(agent)) + .Build(); +app.Run(); +``` + +## Feedback & Contributing + +We welcome feedback and contributions in [our GitHub repo](https://github.com/microsoft/agent-framework). diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/ChatCompletions/Converters/ChatClientAgentRunOptionsConverter.cs b/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/ChatCompletions/Converters/ChatClientAgentRunOptionsConverter.cs index 5f50251f74..3158d87848 100644 --- a/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/ChatCompletions/Converters/ChatClientAgentRunOptionsConverter.cs +++ b/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/ChatCompletions/Converters/ChatClientAgentRunOptionsConverter.cs @@ -10,7 +10,7 @@ namespace Microsoft.Agents.AI.Hosting.OpenAI.ChatCompletions.Converters; internal static class ChatClientAgentRunOptionsConverter { - private static readonly JsonElement s_emptyJson = JsonDocument.Parse("{}").RootElement; + private static readonly JsonElement s_emptyJson = JsonElement.Parse("{}"); public static ChatClientAgentRunOptions BuildOptions(this CreateChatCompletion request) { diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/ChatCompletions/Models/Tool.cs b/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/ChatCompletions/Models/Tool.cs index 470f7d15b0..412494eeaa 100644 --- a/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/ChatCompletions/Models/Tool.cs +++ b/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/ChatCompletions/Models/Tool.cs @@ -18,7 +18,7 @@ internal abstract record Tool /// /// The type of the tool. /// - [JsonPropertyName("type")] + [JsonIgnore] public abstract string Type { get; } } @@ -30,7 +30,7 @@ internal sealed record FunctionTool : Tool /// /// The type of the tool. Always "function". /// - [JsonPropertyName("type")] + [JsonIgnore] public override string Type => "function"; /// @@ -88,7 +88,7 @@ internal sealed record CustomTool : Tool /// /// The type of the tool. Always "custom". /// - [JsonPropertyName("type")] + [JsonIgnore] public override string Type => "custom"; /// @@ -160,5 +160,5 @@ internal sealed record CustomToolFormat /// Additional format properties (schema definition). /// [JsonExtensionData] - public Dictionary? AdditionalProperties { get; init; } + public Dictionary? AdditionalProperties { get; set; } } diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Conversations/InMemoryConversationStorage.cs b/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Conversations/InMemoryConversationStorage.cs index 11b9dd9f0a..d537f33eb9 100644 --- a/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Conversations/InMemoryConversationStorage.cs +++ b/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Conversations/InMemoryConversationStorage.cs @@ -210,11 +210,10 @@ private sealed class ConversationState #if NET9_0_OR_GREATER private readonly OrderedDictionary _items = []; private readonly object _lock = new(); - private Conversation _conversation; public ConversationState(Conversation conversation) { - this._conversation = conversation; + this.Conversation = conversation; } public Conversation Conversation @@ -223,16 +222,18 @@ public Conversation Conversation { lock (this._lock) { - return this._conversation; + return field; } } + + private set; } public void UpdateConversation(Conversation conversation) { lock (this._lock) { - this._conversation = conversation; + this.Conversation = conversation; } } @@ -274,11 +275,10 @@ public bool RemoveItem(string itemId) #else private readonly List _items = []; private readonly object _lock = new(); - private Conversation _conversation; public ConversationState(Conversation conversation) { - this._conversation = conversation; + this.Conversation = conversation; } public Conversation Conversation @@ -287,16 +287,18 @@ public Conversation Conversation { lock (this._lock) { - return this._conversation; + return field; } } + + private set; } public void UpdateConversation(Conversation conversation) { lock (this._lock) { - this._conversation = conversation; + this.Conversation = conversation; } } diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/EndpointRouteBuilderExtensions.Responses.cs b/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/EndpointRouteBuilderExtensions.Responses.cs index f4159011cf..9a395b9b12 100644 --- a/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/EndpointRouteBuilderExtensions.Responses.cs +++ b/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/EndpointRouteBuilderExtensions.Responses.cs @@ -3,6 +3,7 @@ using System; using System.Diagnostics.CodeAnalysis; using Microsoft.Agents.AI; +using Microsoft.Agents.AI.Hosting; using Microsoft.Agents.AI.Hosting.OpenAI; using Microsoft.Agents.AI.Hosting.OpenAI.Conversations; using Microsoft.Agents.AI.Hosting.OpenAI.Responses; @@ -17,6 +18,29 @@ namespace Microsoft.AspNetCore.Builder; /// public static partial class MicrosoftAgentAIHostingOpenAIEndpointRouteBuilderExtensions { + /// + /// Maps OpenAI Responses API endpoints to the specified for the given . + /// + /// The to add the OpenAI Responses endpoints to. + /// The builder for to map the OpenAI Responses endpoints for. + public static IEndpointConventionBuilder MapOpenAIResponses(this IEndpointRouteBuilder endpoints, IHostedAgentBuilder agentBuilder) + => MapOpenAIResponses(endpoints, agentBuilder, path: null); + + /// + /// Maps OpenAI Responses API endpoints to the specified for the given . + /// + /// The to add the OpenAI Responses endpoints to. + /// The builder for to map the OpenAI Responses endpoints for. + /// Custom route path for the OpenAI Responses endpoint. + public static IEndpointConventionBuilder MapOpenAIResponses(this IEndpointRouteBuilder endpoints, IHostedAgentBuilder agentBuilder, string? path) + { + ArgumentNullException.ThrowIfNull(endpoints); + ArgumentNullException.ThrowIfNull(agentBuilder); + + var agent = endpoints.ServiceProvider.GetRequiredKeyedService(agentBuilder.Name); + return MapOpenAIResponses(endpoints, agent, path); + } + /// /// Maps OpenAI Responses API endpoints to the specified for the given . /// diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/IdGenerator.cs b/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/IdGenerator.cs index bd35fa8308..5741e8d161 100644 --- a/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/IdGenerator.cs +++ b/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/IdGenerator.cs @@ -146,6 +146,9 @@ private static string GetRandomString(int stringLength, Random? random) const string Chars = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789"; if (random is not null) { +#if NET10_0_OR_GREATER + return random.GetString(Chars, stringLength); +#else // Use deterministic random generation when seed is provided return string.Create(stringLength, random, static (destination, random) => { @@ -154,6 +157,7 @@ private static string GetRandomString(int stringLength, Random? random) destination[i] = Chars[random.Next(Chars.Length)]; } }); +#endif } // Use cryptographically secure random generation when no seed is provided diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Microsoft.Agents.AI.Hosting.OpenAI.csproj b/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Microsoft.Agents.AI.Hosting.OpenAI.csproj index 707cc4fe68..923f8e3eb6 100644 --- a/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Microsoft.Agents.AI.Hosting.OpenAI.csproj +++ b/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Microsoft.Agents.AI.Hosting.OpenAI.csproj @@ -1,8 +1,7 @@  - $(ProjectsCoreTargetFrameworks) - $(ProjectsDebugCoreTargetFrameworks) + $(TargetFrameworksCore) $(NoWarn);OPENAI001;MEAI001 Microsoft.Agents.AI.Hosting.OpenAI alpha @@ -22,12 +21,13 @@ - + + + - - + diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/OpenAIHostingJsonUtilities.cs b/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/OpenAIHostingJsonUtilities.cs index 49ceef622a..f77143c583 100644 --- a/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/OpenAIHostingJsonUtilities.cs +++ b/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/OpenAIHostingJsonUtilities.cs @@ -109,6 +109,7 @@ private static JsonSerializerOptions CreateDefaultOptions() [JsonSerializable(typeof(MCPApprovalRequestItemResource))] [JsonSerializable(typeof(MCPApprovalResponseItemResource))] [JsonSerializable(typeof(MCPCallItemResource))] +[JsonSerializable(typeof(ExecutorActionItemResource))] [JsonSerializable(typeof(List))] // ItemParam types [JsonSerializable(typeof(ItemParam))] diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/AIAgentResponseExecutor.cs b/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/AIAgentResponseExecutor.cs index 18863034bf..e3706bee1c 100644 --- a/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/AIAgentResponseExecutor.cs +++ b/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/AIAgentResponseExecutor.cs @@ -24,6 +24,10 @@ public AIAgentResponseExecutor(AIAgent agent) this._agent = agent; } + public ValueTask ValidateRequestAsync( + CreateResponse request, + CancellationToken cancellationToken = default) => ValueTask.FromResult(null); + public async IAsyncEnumerable ExecuteAsync( AgentInvocationContext context, CreateResponse request, @@ -32,7 +36,13 @@ public async IAsyncEnumerable ExecuteAsync( // Create options with properties from the request var chatOptions = new ChatOptions { - ConversationId = request.Conversation?.Id, + // Note: We intentionally do NOT set ConversationId on ChatOptions here. + // The conversation ID from the client request is used by the hosting layer + // to manage conversation storage, but should not be forwarded to the underlying + // IChatClient as it has its own concept of conversations (or none at all). + // --- + // ConversationId = request.Conversation?.Id, + Temperature = (float?)request.Temperature, TopP = (float?)request.TopP, MaxOutputTokens = request.MaxOutputTokens, diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/AgentRunResponseExtensions.cs b/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/AgentRunResponseExtensions.cs index fedaeae1f4..97dcf9740f 100644 --- a/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/AgentRunResponseExtensions.cs +++ b/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/AgentRunResponseExtensions.cs @@ -56,7 +56,7 @@ public static Response ToResponse( MaxOutputTokens = request.MaxOutputTokens, MaxToolCalls = request.MaxToolCalls, Metadata = request.Metadata is IReadOnlyDictionary metadata ? new Dictionary(metadata) : [], - Model = request.Agent?.Name ?? request.Model, + Model = request.Model, Output = output, ParallelToolCalls = request.ParallelToolCalls ?? true, PreviousResponseId = request.PreviousResponseId, @@ -64,7 +64,7 @@ public static Response ToResponse( PromptCacheKey = request.PromptCacheKey, Reasoning = request.Reasoning, SafetyIdentifier = request.SafetyIdentifier, - ServiceTier = request.ServiceTier ?? "default", + ServiceTier = request.ServiceTier, Status = ResponseStatus.Completed, Store = request.Store ?? true, Temperature = request.Temperature ?? 1.0, diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/AgentRunResponseUpdateExtensions.cs b/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/AgentRunResponseUpdateExtensions.cs index 252cdc8d92..628b80b340 100644 --- a/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/AgentRunResponseUpdateExtensions.cs +++ b/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/AgentRunResponseUpdateExtensions.cs @@ -45,6 +45,9 @@ public static async IAsyncEnumerable ToStreamingResponse var updateEnumerator = updates.GetAsyncEnumerator(cancellationToken); await using var _ = updateEnumerator.ConfigureAwait(false); + // Track active item IDs by executor ID to pair invoked/completed/failed events + Dictionary executorItemIds = []; + AgentRunResponseUpdate? previousUpdate = null; StreamingEventGenerator? generator = null; while (await updateEnumerator.MoveNextAsync().ConfigureAwait(false)) @@ -55,7 +58,92 @@ public static async IAsyncEnumerable ToStreamingResponse // Special-case for agent framework workflow events. if (update.RawRepresentation is WorkflowEvent workflowEvent) { - yield return CreateWorkflowEventResponse(workflowEvent, seq.Increment(), outputIndex); + // Convert executor events to standard OpenAI output_item events + if (workflowEvent is ExecutorInvokedEvent invokedEvent) + { + var itemId = IdGenerator.NewId(prefix: "item"); + // Store the item ID for this executor so we can reuse it for completion/failure + executorItemIds[invokedEvent.ExecutorId] = itemId; + + var item = new ExecutorActionItemResource + { + Id = itemId, + ExecutorId = invokedEvent.ExecutorId, + Status = "in_progress", + CreatedAt = DateTimeOffset.UtcNow.ToUnixTimeSeconds() + }; + + yield return new StreamingOutputItemAdded + { + SequenceNumber = seq.Increment(), + OutputIndex = outputIndex, + Item = item + }; + } + else if (workflowEvent is ExecutorCompletedEvent completedEvent) + { + // Reuse the item ID from the invoked event, or generate a new one if not found + var itemId = executorItemIds.TryGetValue(completedEvent.ExecutorId, out var existingId) + ? existingId + : IdGenerator.NewId(prefix: "item"); + + // Remove from tracking as this executor run is now complete + executorItemIds.Remove(completedEvent.ExecutorId); + JsonElement? resultData = null; + if (completedEvent.Data != null && JsonSerializer.IsReflectionEnabledByDefault) + { + resultData = JsonSerializer.SerializeToElement( + completedEvent.Data, + OpenAIHostingJsonUtilities.DefaultOptions.GetTypeInfo(typeof(object))); + } + + var item = new ExecutorActionItemResource + { + Id = itemId, + ExecutorId = completedEvent.ExecutorId, + Status = "completed", + Result = resultData, + CreatedAt = DateTimeOffset.UtcNow.ToUnixTimeSeconds() + }; + + yield return new StreamingOutputItemDone + { + SequenceNumber = seq.Increment(), + OutputIndex = outputIndex, + Item = item + }; + } + else if (workflowEvent is ExecutorFailedEvent failedEvent) + { + // Reuse the item ID from the invoked event, or generate a new one if not found + var itemId = executorItemIds.TryGetValue(failedEvent.ExecutorId, out var existingId) + ? existingId + : IdGenerator.NewId(prefix: "item"); + + // Remove from tracking as this executor run has now failed + executorItemIds.Remove(failedEvent.ExecutorId); + + var item = new ExecutorActionItemResource + { + Id = itemId, + ExecutorId = failedEvent.ExecutorId, + Status = "failed", + Error = failedEvent.Data?.ToString(), + CreatedAt = DateTimeOffset.UtcNow.ToUnixTimeSeconds() + }; + + yield return new StreamingOutputItemDone + { + SequenceNumber = seq.Increment(), + OutputIndex = outputIndex, + Item = item + }; + } + else + { + // For other workflow events (not executor-specific), keep the old format as fallback + yield return CreateWorkflowEventResponse(workflowEvent, seq.Increment(), outputIndex); + } continue; } @@ -165,7 +253,7 @@ Response CreateResponse(ResponseStatus status = ResponseStatus.Completed, IEnume MaxOutputTokens = request.MaxOutputTokens, MaxToolCalls = request.MaxToolCalls, Metadata = request.Metadata != null ? new Dictionary(request.Metadata) : [], - Model = request.Agent?.Name ?? request.Model, + Model = request.Model, Output = outputs?.ToList() ?? [], ParallelToolCalls = request.ParallelToolCalls ?? true, PreviousResponseId = request.PreviousResponseId, diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/Converters/ItemContentConverter.cs b/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/Converters/ItemContentConverter.cs index 32262d2e2c..2476ce2fbd 100644 --- a/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/Converters/ItemContentConverter.cs +++ b/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/Converters/ItemContentConverter.cs @@ -140,10 +140,7 @@ DataContent audioData when audioData.HasTopLevelMediaType("audio") => _ => null }; - if (result is not null) - { - result.RawRepresentation = content; - } + result?.RawRepresentation = content; return result; } diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/Converters/ItemResourceConverter.cs b/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/Converters/ItemResourceConverter.cs index 571e45fa1f..0ca5c05d9b 100644 --- a/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/Converters/ItemResourceConverter.cs +++ b/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/Converters/ItemResourceConverter.cs @@ -45,6 +45,7 @@ internal sealed class ItemResourceConverter : JsonConverter MCPApprovalRequestItemResource.ItemType => doc.Deserialize(OpenAIHostingJsonContext.Default.MCPApprovalRequestItemResource), MCPApprovalResponseItemResource.ItemType => doc.Deserialize(OpenAIHostingJsonContext.Default.MCPApprovalResponseItemResource), MCPCallItemResource.ItemType => doc.Deserialize(OpenAIHostingJsonContext.Default.MCPCallItemResource), + ExecutorActionItemResource.ItemType => doc.Deserialize(OpenAIHostingJsonContext.Default.ExecutorActionItemResource), _ => null }; } @@ -106,6 +107,9 @@ public override void Write(Utf8JsonWriter writer, ItemResource value, JsonSerial case MCPCallItemResource mcpCall: JsonSerializer.Serialize(writer, mcpCall, OpenAIHostingJsonContext.Default.MCPCallItemResource); break; + case ExecutorActionItemResource executorAction: + JsonSerializer.Serialize(writer, executorAction, OpenAIHostingJsonContext.Default.ExecutorActionItemResource); + break; default: throw new JsonException($"Unknown item type: {value.GetType().Name}"); } diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/HostedAgentResponseExecutor.cs b/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/HostedAgentResponseExecutor.cs index 78e4331b6b..78cf89b970 100644 --- a/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/HostedAgentResponseExecutor.cs +++ b/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/HostedAgentResponseExecutor.cs @@ -13,8 +13,9 @@ namespace Microsoft.Agents.AI.Hosting.OpenAI.Responses; /// -/// Response executor that routes requests to hosted AIAgent services based on the model or agent.name parameter. +/// Response executor that routes requests to hosted AIAgent services based on agent.name or metadata["entity_id"]. /// This executor resolves agents from keyed services registered via AddAIAgent(). +/// The model field is reserved for actual model names and is never used for entity/agent identification. /// internal sealed class HostedAgentResponseExecutor : IResponseExecutor { @@ -37,19 +38,64 @@ public HostedAgentResponseExecutor( this._logger = logger; } + /// + public ValueTask ValidateRequestAsync( + CreateResponse request, + CancellationToken cancellationToken = default) + { + // Extract agent name from agent.name or model parameter + string? agentName = GetAgentName(request); + + if (string.IsNullOrEmpty(agentName)) + { + return ValueTask.FromResult(new ResponseError + { + Code = "missing_required_parameter", + Message = "No 'agent.name' or 'metadata[\"entity_id\"]' specified in the request." + }); + } + + // Validate that the agent can be resolved + AIAgent? agent = this._serviceProvider.GetKeyedService(agentName); + if (agent is null) + { + if (this._logger.IsEnabled(LogLevel.Warning)) + { + this._logger.LogWarning("Failed to resolve agent with name '{AgentName}'", agentName); + } + + return ValueTask.FromResult(new ResponseError + { + Code = "agent_not_found", + Message = $""" + Agent '{agentName}' not found. + Ensure the agent is registered with '{agentName}' name in the dependency injection container. + We recommend using 'builder.AddAIAgent()' for simplicity. + """ + }); + } + + return ValueTask.FromResult(null); + } + /// public async IAsyncEnumerable ExecuteAsync( AgentInvocationContext context, CreateResponse request, [EnumeratorCancellation] CancellationToken cancellationToken = default) { - // Validate and resolve agent synchronously to ensure validation errors are thrown immediately - AIAgent agent = this.ResolveAgent(request); + string agentName = GetAgentName(request)!; + AIAgent agent = this._serviceProvider.GetRequiredKeyedService(agentName); - // Create options with properties from the request var chatOptions = new ChatOptions { - ConversationId = request.Conversation?.Id, + // Note: We intentionally do NOT set ConversationId on ChatOptions here. + // The conversation ID from the client request is used by the hosting layer + // to manage conversation storage, but should not be forwarded to the underlying + // IChatClient as it has its own concept of conversations (or none at all). + // --- + // ConversationId = request.Conversation?.Id, + Temperature = (float?)request.Temperature, TopP = (float?)request.TopP, MaxOutputTokens = request.MaxOutputTokens, @@ -57,8 +103,6 @@ public async IAsyncEnumerable ExecuteAsync( ModelId = request.Model, }; var options = new ChatClientAgentRunOptions(chatOptions); - - // Convert input to chat messages var messages = new List(); foreach (var inputMessage in request.Input.GetInputMessages()) @@ -66,7 +110,6 @@ public async IAsyncEnumerable ExecuteAsync( messages.Add(inputMessage.ToChatMessage()); } - // Use the extension method to convert streaming updates to streaming response events await foreach (var streamingEvent in agent.RunStreamingAsync(messages, options: options, cancellationToken: cancellationToken) .ToStreamingResponseAsync(request, context, cancellationToken).ConfigureAwait(false)) { @@ -75,41 +118,20 @@ public async IAsyncEnumerable ExecuteAsync( } /// - /// Resolves an agent from the service provider based on the request. + /// Extracts the agent name for a request from the agent.name property, falling back to metadata["entity_id"]. /// /// The create response request. - /// The resolved AIAgent instance. - /// Thrown when the agent cannot be resolved. - private AIAgent ResolveAgent(CreateResponse request) + /// The agent name. + private static string? GetAgentName(CreateResponse request) { - // Extract agent name from agent.name or model parameter - var agentName = request.Agent?.Name ?? request.Model; - if (string.IsNullOrEmpty(agentName)) - { - throw new InvalidOperationException("No 'agent.name' or 'model' specified in the request."); - } + string? agentName = request.Agent?.Name; - // Resolve the keyed agent service - try + // Fall back to metadata["entity_id"] if agent.name is not present + if (string.IsNullOrEmpty(agentName) && request.Metadata?.TryGetValue("entity_id", out string? entityId) == true) { - return this._serviceProvider.GetRequiredKeyedService(agentName); + agentName = entityId; } - catch (InvalidOperationException ex) - { - this._logger.LogError(ex, "Failed to resolve agent with name '{AgentName}'", agentName); - throw new InvalidOperationException($"Agent '{agentName}' not found. Ensure the agent is registered with AddAIAgent().", ex); - } - } - /// - /// Validates that the agent can be resolved without actually resolving it. - /// This allows early validation before starting async execution. - /// - /// The create response request. - /// Thrown when the agent cannot be resolved. - public void ValidateAgent(CreateResponse request) - { - // Use the same logic as ResolveAgent but don't return the agent - _ = this.ResolveAgent(request); + return agentName; } } diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/IResponseExecutor.cs b/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/IResponseExecutor.cs index ca4da70b88..b96879f4cc 100644 --- a/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/IResponseExecutor.cs +++ b/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/IResponseExecutor.cs @@ -2,6 +2,7 @@ using System.Collections.Generic; using System.Threading; +using System.Threading.Tasks; using Microsoft.Agents.AI.Hosting.OpenAI.Responses.Models; namespace Microsoft.Agents.AI.Hosting.OpenAI.Responses; @@ -12,6 +13,16 @@ namespace Microsoft.Agents.AI.Hosting.OpenAI.Responses; /// internal interface IResponseExecutor { + /// + /// Validates a create response request before execution. + /// + /// The create response request to validate. + /// Cancellation token. + /// A if validation fails, null if validation succeeds. + ValueTask ValidateRequestAsync( + CreateResponse request, + CancellationToken cancellationToken = default); + /// /// Executes a response generation request and returns streaming events. /// diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/IResponsesService.cs b/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/IResponsesService.cs index 67f7b72f20..b1676ac99c 100644 --- a/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/IResponsesService.cs +++ b/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/IResponsesService.cs @@ -18,6 +18,17 @@ internal interface IResponsesService /// Default limit for list operations. /// const int DefaultListLimit = 20; + + /// + /// Validates a create response request before execution. + /// + /// The create response request to validate. + /// Cancellation token. + /// A ResponseError if validation fails, null if validation succeeds. + ValueTask ValidateRequestAsync( + CreateResponse request, + CancellationToken cancellationToken = default); + /// /// Creates a model response for the given input. /// diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/InMemoryResponsesService.cs b/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/InMemoryResponsesService.cs index dfb744596a..2f5b3f4660 100644 --- a/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/InMemoryResponsesService.cs +++ b/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/InMemoryResponsesService.cs @@ -147,18 +147,27 @@ public InMemoryResponsesService(IResponseExecutor executor, InMemoryStorageOptio this._conversationStorage = conversationStorage; } - public async Task CreateResponseAsync( + public async ValueTask ValidateRequestAsync( CreateResponse request, CancellationToken cancellationToken = default) { - ValidateRequest(request); - - // Validate agent resolution early for HostedAgentResponseExecutor - if (this._executor is HostedAgentResponseExecutor hostedExecutor) + if (request.Conversation is not null && !string.IsNullOrEmpty(request.Conversation.Id) && + !string.IsNullOrEmpty(request.PreviousResponseId)) { - hostedExecutor.ValidateAgent(request); + return new ResponseError + { + Code = "invalid_request", + Message = "Mutually exclusive parameters: 'conversation' and 'previous_response_id'. Ensure you are only providing one of: 'previous_response_id' or 'conversation'." + }; } + return await this._executor.ValidateRequestAsync(request, cancellationToken).ConfigureAwait(false); + } + + public async Task CreateResponseAsync( + CreateResponse request, + CancellationToken cancellationToken = default) + { if (request.Stream == true) { throw new InvalidOperationException("Cannot create a streaming response using CreateResponseAsync. Use CreateResponseStreamingAsync instead."); @@ -189,8 +198,6 @@ public async IAsyncEnumerable CreateResponseStreamingAsy CreateResponse request, [EnumeratorCancellation] CancellationToken cancellationToken = default) { - ValidateRequest(request); - if (request.Stream == false) { throw new InvalidOperationException("Cannot create a non-streaming response using CreateResponseStreamingAsync. Use CreateResponseAsync instead."); @@ -342,15 +349,6 @@ public Task> ListResponseInputItemsAsync( }); } - private static void ValidateRequest(CreateResponse request) - { - if (request.Conversation is not null && !string.IsNullOrEmpty(request.Conversation.Id) && - !string.IsNullOrEmpty(request.PreviousResponseId)) - { - throw new InvalidOperationException("Mutually exclusive parameters: 'conversation' and 'previous_response_id'. Ensure you are only providing one of: 'previous_response_id' or 'conversation'."); - } - } - private ResponseState InitializeResponse(string responseId, CreateResponse request) { var metadata = request.Metadata ?? []; @@ -371,7 +369,7 @@ private ResponseState InitializeResponse(string responseId, CreateResponse reque MaxOutputTokens = request.MaxOutputTokens, MaxToolCalls = request.MaxToolCalls, Metadata = metadata, - Model = request.Model ?? "default", + Model = request.Model, Output = [], ParallelToolCalls = request.ParallelToolCalls ?? true, PreviousResponseId = request.PreviousResponseId, diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/Models/InputMessage.cs b/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/Models/InputMessage.cs index 029be0752a..c1ede61188 100644 --- a/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/Models/InputMessage.cs +++ b/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/Models/InputMessage.cs @@ -40,7 +40,7 @@ public ChatMessage ToChatMessage() { if (this.Content.IsText) { - return new ChatMessage(this.Role, this.Content.Text!); + return new ChatMessage(this.Role, this.Content.Text); } else if (this.Content.IsContents) { diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/Models/ItemResource.cs b/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/Models/ItemResource.cs index 0a543e1be9..289bafbc43 100644 --- a/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/Models/ItemResource.cs +++ b/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/Models/ItemResource.cs @@ -888,3 +888,47 @@ internal sealed class MCPCallItemResource : ItemResource [JsonPropertyName("error")] public string? Error { get; init; } } + +/// +/// An executor action item resource for workflow execution visualization. +/// +internal sealed class ExecutorActionItemResource : ItemResource +{ + /// + /// The constant item type identifier for executor action items. + /// + public const string ItemType = "executor_action"; + + /// + public override string Type => ItemType; + + /// + /// The executor identifier. + /// + [JsonPropertyName("executor_id")] + public required string ExecutorId { get; init; } + + /// + /// The execution status: "in_progress", "completed", "failed", or "cancelled". + /// + [JsonPropertyName("status")] + public required string Status { get; init; } + + /// + /// The executor result data (for completed status). + /// + [JsonPropertyName("result")] + public JsonElement? Result { get; init; } + + /// + /// The error message (for failed status). + /// + [JsonPropertyName("error")] + public string? Error { get; init; } + + /// + /// The creation timestamp. + /// + [JsonPropertyName("created_at")] + public long CreatedAt { get; init; } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/Models/ResponseInput.cs b/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/Models/ResponseInput.cs index d0555a2c00..d291b93528 100644 --- a/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/Models/ResponseInput.cs +++ b/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/Models/ResponseInput.cs @@ -182,7 +182,9 @@ internal sealed class ResponseInputJsonConverter : JsonConverter return messages is not null ? ResponseInput.FromMessages(messages) : null; } - throw new JsonException($"Unexpected token type for ResponseInput: {reader.TokenType}"); + throw new JsonException( + "ResponseInput must be either a string or an array of messages. " + + $"Objects are not supported. Received token type: {reader.TokenType}"); } /// diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/Models/StreamingResponseEvent.cs b/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/Models/StreamingResponseEvent.cs index 6d41e10aff..f39c6e4bca 100644 --- a/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/Models/StreamingResponseEvent.cs +++ b/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/Models/StreamingResponseEvent.cs @@ -565,7 +565,7 @@ internal sealed class StreamingWorkflowEventComplete : StreamingResponseEvent /// /// The constant event type identifier for workflow event events. /// - public const string EventType = "response.workflow_event.complete"; + public const string EventType = "response.workflow_event.completed"; /// [JsonIgnore] diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/ResponsesHttpHandler.cs b/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/ResponsesHttpHandler.cs index 31f61e967e..b73cdebda5 100644 --- a/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/ResponsesHttpHandler.cs +++ b/dotnet/src/Microsoft.Agents.AI.Hosting.OpenAI/Responses/ResponsesHttpHandler.cs @@ -34,6 +34,21 @@ public async Task CreateResponseAsync( [FromQuery] bool? stream, CancellationToken cancellationToken) { + // Validate the request first + ResponseError? validationError = await this._responsesService.ValidateRequestAsync(request, cancellationToken).ConfigureAwait(false); + if (validationError is not null) + { + return Results.BadRequest(new ErrorResponse + { + Error = new ErrorDetails + { + Message = validationError.Message, + Type = "invalid_request_error", + Code = validationError.Code + } + }); + } + try { // Handle streaming vs non-streaming @@ -55,45 +70,24 @@ public async Task CreateResponseAsync( request, cancellationToken: cancellationToken).ConfigureAwait(false); - return Results.Ok(response); - } - catch (InvalidOperationException ex) when (ex.Message.Contains("Mutually exclusive")) - { - // Return OpenAI-style error for mutual exclusivity violations - return Results.BadRequest(new ErrorResponse + return response.Status switch { - Error = new ErrorDetails - { - Message = ex.Message, - Type = "invalid_request_error", - Code = "mutually_exclusive_parameters" - } - }); - } - catch (InvalidOperationException ex) when (ex.Message.Contains("not found") || ex.Message.Contains("does not exist")) - { - // Return OpenAI-style error for not found errors - return Results.NotFound(new ErrorResponse - { - Error = new ErrorDetails - { - Message = ex.Message, - Type = "invalid_request_error" - } - }); + ResponseStatus.Failed when response.Error is { } error => Results.Problem( + detail: error.Message, + statusCode: StatusCodes.Status500InternalServerError, + title: error.Code ?? "Internal Server Error"), + ResponseStatus.Failed => Results.Problem(), + ResponseStatus.Queued => Results.Accepted(value: response), + _ => Results.Ok(response) + }; } - catch (InvalidOperationException ex) when (ex.Message.Contains("No 'agent.name' or 'model' specified")) + catch (Exception ex) { - // Return OpenAI-style error for missing required parameters - return Results.BadRequest(new ErrorResponse - { - Error = new ErrorDetails - { - Message = ex.Message, - Type = "invalid_request_error", - Code = "missing_required_parameter" - } - }); + // Return InternalServerError for unexpected exceptions + return Results.Problem( + detail: ex.Message, + statusCode: StatusCodes.Status500InternalServerError, + title: "Internal Server Error"); } } diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting/AgentCatalog.cs b/dotnet/src/Microsoft.Agents.AI.Hosting/AgentCatalog.cs deleted file mode 100644 index 0d2ef69640..0000000000 --- a/dotnet/src/Microsoft.Agents.AI.Hosting/AgentCatalog.cs +++ /dev/null @@ -1,38 +0,0 @@ -// Copyright (c) Microsoft. All rights reserved. - -using System.Collections.Generic; -using System.Threading; - -namespace Microsoft.Agents.AI.Hosting; - -/// -/// Provides a catalog of registered AI agents within the hosting environment. -/// -/// -/// The agent catalog allows enumeration of all registered agents in the dependency injection container. -/// This is useful for scenarios where you need to discover and interact with multiple agents programmatically. -/// -public abstract class AgentCatalog -{ - /// - /// Initializes a new instance of the class. - /// - protected AgentCatalog() - { - } - - /// - /// Asynchronously retrieves all registered AI agents from the catalog. - /// - /// The to monitor for cancellation requests. The default is . - /// - /// An asynchronous enumerable of instances representing all registered agents. - /// The enumeration will only include agents that are successfully resolved from the service provider. - /// - /// - /// This method enumerates through all registered agent names and attempts to resolve each agent - /// from the dependency injection container. Only successfully resolved agents are yielded. - /// The enumeration is lazy and agents are resolved on-demand during iteration. - /// - public abstract IAsyncEnumerable GetAgentsAsync(CancellationToken cancellationToken = default); -} diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting/AgentHostingServiceCollectionExtensions.cs b/dotnet/src/Microsoft.Agents.AI.Hosting/AgentHostingServiceCollectionExtensions.cs index 3c5a5b84c2..e12d017343 100644 --- a/dotnet/src/Microsoft.Agents.AI.Hosting/AgentHostingServiceCollectionExtensions.cs +++ b/dotnet/src/Microsoft.Agents.AI.Hosting/AgentHostingServiceCollectionExtensions.cs @@ -1,7 +1,7 @@ // Copyright (c) Microsoft. All rights reserved. using System; -using System.Linq; +using System.Collections.Generic; using Microsoft.Agents.AI.Hosting.Local; using Microsoft.Extensions.AI; using Microsoft.Extensions.DependencyInjection; @@ -29,7 +29,8 @@ public static IHostedAgentBuilder AddAIAgent(this IServiceCollection services, s return services.AddAIAgent(name, (sp, key) => { var chatClient = sp.GetRequiredService(); - return new ChatClientAgent(chatClient, instructions, key); + var tools = GetRegisteredToolsForAgent(sp, name); + return new ChatClientAgent(chatClient, instructions, key, tools: tools); }); } @@ -46,7 +47,11 @@ public static IHostedAgentBuilder AddAIAgent(this IServiceCollection services, s { Throw.IfNull(services); Throw.IfNullOrEmpty(name); - return services.AddAIAgent(name, (sp, key) => new ChatClientAgent(chatClient, instructions, key)); + return services.AddAIAgent(name, (sp, key) => + { + var tools = GetRegisteredToolsForAgent(sp, name); + return new ChatClientAgent(chatClient, instructions, key, tools: tools); + }); } /// @@ -65,7 +70,8 @@ public static IHostedAgentBuilder AddAIAgent(this IServiceCollection services, s return services.AddAIAgent(name, (sp, key) => { var chatClient = chatClientServiceKey is null ? sp.GetRequiredService() : sp.GetRequiredKeyedService(chatClientServiceKey); - return new ChatClientAgent(chatClient, instructions, key); + var tools = GetRegisteredToolsForAgent(sp, name); + return new ChatClientAgent(chatClient, instructions, key, tools: tools); }); } @@ -86,7 +92,8 @@ public static IHostedAgentBuilder AddAIAgent(this IServiceCollection services, s return services.AddAIAgent(name, (sp, key) => { var chatClient = chatClientServiceKey is null ? sp.GetRequiredService() : sp.GetRequiredKeyedService(chatClientServiceKey); - return new ChatClientAgent(chatClient, instructions: instructions, name: key, description: description); + var tools = GetRegisteredToolsForAgent(sp, name); + return new ChatClientAgent(chatClient, instructions: instructions, name: key, description: description, tools: tools); }); } @@ -118,28 +125,12 @@ public static IHostedAgentBuilder AddAIAgent(this IServiceCollection services, s return agent; }); - // Register the agent by name for discovery. - var agentHostBuilder = GetAgentRegistry(services); - agentHostBuilder.AgentNames.Add(name); - return new HostedAgentBuilder(name, services); } - private static LocalAgentRegistry GetAgentRegistry(IServiceCollection services) - { - var descriptor = services.FirstOrDefault(s => !s.IsKeyedService && s.ServiceType.Equals(typeof(LocalAgentRegistry))); - if (descriptor?.ImplementationInstance is not LocalAgentRegistry instance) - { - instance = new LocalAgentRegistry(); - ConfigureHostBuilder(services, instance); - } - - return instance; - } - - private static void ConfigureHostBuilder(IServiceCollection services, LocalAgentRegistry agentHostBuilderContext) + private static IList GetRegisteredToolsForAgent(IServiceProvider serviceProvider, string agentName) { - services.Add(ServiceDescriptor.Singleton(agentHostBuilderContext)); - services.AddSingleton(); + var registry = serviceProvider.GetService(); + return registry?.GetTools(agentName) ?? []; } } diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting/HostApplicationBuilderWorkflowExtensions.cs b/dotnet/src/Microsoft.Agents.AI.Hosting/HostApplicationBuilderWorkflowExtensions.cs index ac78877682..8075caec59 100644 --- a/dotnet/src/Microsoft.Agents.AI.Hosting/HostApplicationBuilderWorkflowExtensions.cs +++ b/dotnet/src/Microsoft.Agents.AI.Hosting/HostApplicationBuilderWorkflowExtensions.cs @@ -1,9 +1,6 @@ // Copyright (c) Microsoft. All rights reserved. using System; -using System.Collections.Generic; -using System.Linq; -using Microsoft.Agents.AI.Hosting.Local; using Microsoft.Agents.AI.Workflows; using Microsoft.Extensions.DependencyInjection; using Microsoft.Extensions.Hosting; @@ -16,46 +13,6 @@ namespace Microsoft.Agents.AI.Hosting; /// public static class HostApplicationBuilderWorkflowExtensions { - /// - /// Registers a concurrent workflow that executes multiple agents in parallel. - /// - /// The to configure. - /// The unique name for the workflow. - /// A collection of instances representing agents to execute concurrently. - /// An that can be used to further configure the workflow. - /// Thrown when , , or is null. - /// Thrown when or is empty. - public static IHostedWorkflowBuilder AddConcurrentWorkflow(this IHostApplicationBuilder builder, string name, IEnumerable agentBuilders) - { - Throw.IfNullOrEmpty(agentBuilders); - - return builder.AddWorkflow(name, (sp, key) => - { - var agents = agentBuilders.Select(ab => sp.GetRequiredKeyedService(ab.Name)); - return AgentWorkflowBuilder.BuildConcurrent(workflowName: name, agents: agents); - }); - } - - /// - /// Registers a sequential workflow that executes agents in a specific order. - /// - /// The to configure. - /// The unique name for the workflow. - /// A collection of instances representing agents to execute in sequence. - /// An that can be used to further configure the workflow. - /// Thrown when , , or is null. - /// Thrown when or is empty. - public static IHostedWorkflowBuilder AddSequentialWorkflow(this IHostApplicationBuilder builder, string name, IEnumerable agentBuilders) - { - Throw.IfNullOrEmpty(agentBuilders); - - return builder.AddWorkflow(name, (sp, key) => - { - var agents = agentBuilders.Select(ab => sp.GetRequiredKeyedService(ab.Name)); - return AgentWorkflowBuilder.BuildSequential(workflowName: name, agents: agents); - }); - } - /// /// Registers a custom workflow using a factory delegate. /// @@ -88,28 +45,6 @@ public static IHostedWorkflowBuilder AddWorkflow(this IHostApplicationBuilder bu return workflow; }); - // Register the workflow by name for discovery. - var workflowRegistry = GetWorkflowRegistry(builder); - workflowRegistry.WorkflowNames.Add(name); - return new HostedWorkflowBuilder(name, builder); } - - private static LocalWorkflowRegistry GetWorkflowRegistry(IHostApplicationBuilder builder) - { - var descriptor = builder.Services.FirstOrDefault(s => !s.IsKeyedService && s.ServiceType.Equals(typeof(LocalWorkflowRegistry))); - if (descriptor?.ImplementationInstance is not LocalWorkflowRegistry instance) - { - instance = new LocalWorkflowRegistry(); - ConfigureHostBuilder(builder, instance); - } - - return instance; - } - - private static void ConfigureHostBuilder(IHostApplicationBuilder builder, LocalWorkflowRegistry agentHostBuilderContext) - { - builder.Services.Add(ServiceDescriptor.Singleton(agentHostBuilderContext)); - builder.Services.AddSingleton(); - } } diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting/HostedAgentBuilderExtensions.cs b/dotnet/src/Microsoft.Agents.AI.Hosting/HostedAgentBuilderExtensions.cs index 902c54ebe9..d3a437663a 100644 --- a/dotnet/src/Microsoft.Agents.AI.Hosting/HostedAgentBuilderExtensions.cs +++ b/dotnet/src/Microsoft.Agents.AI.Hosting/HostedAgentBuilderExtensions.cs @@ -1,6 +1,9 @@ // Copyright (c) Microsoft. All rights reserved. using System; +using System.Linq; +using Microsoft.Agents.AI.Hosting.Local; +using Microsoft.Extensions.AI; using Microsoft.Extensions.DependencyInjection; using Microsoft.Shared.Diagnostics; @@ -49,14 +52,57 @@ public static IHostedAgentBuilder WithThreadStore(this IHostedAgentBuilder build Throw.IfNull(key); var keyString = key as string; Throw.IfNullOrEmpty(keyString); - var store = createAgentThreadStore(sp, keyString); - if (store is null) - { + return createAgentThreadStore(sp, keyString) ?? throw new InvalidOperationException($"The agent thread store factory did not return a valid {nameof(AgentThreadStore)} instance for key '{keyString}'."); - } - - return store; }); return builder; } + + /// + /// Adds an AI tool to an agent being configured with the service collection. + /// + /// The hosted agent builder. + /// The AI tool to add to the agent. + /// The same instance so that additional calls can be chained. + /// Thrown when or is . + public static IHostedAgentBuilder WithAITool(this IHostedAgentBuilder builder, AITool tool) + { + Throw.IfNull(builder); + Throw.IfNull(tool); + + var agentName = builder.Name; + var services = builder.ServiceCollection; + + // Get or create the agent tool registry + var descriptor = services.FirstOrDefault(sd => !sd.IsKeyedService && sd.ServiceType.Equals(typeof(LocalAgentToolRegistry))); + if (descriptor?.ImplementationInstance is not LocalAgentToolRegistry toolRegistry) + { + toolRegistry = new(); + services.Add(ServiceDescriptor.Singleton(toolRegistry)); + } + + toolRegistry.AddTool(agentName, tool); + + return builder; + } + + /// + /// Adds multiple AI tools to an agent being configured with the service collection. + /// + /// The hosted agent builder. + /// The collection of AI tools to add to the agent. + /// The same instance so that additional calls can be chained. + /// Thrown when or is . + public static IHostedAgentBuilder WithAITools(this IHostedAgentBuilder builder, params AITool[] tools) + { + Throw.IfNull(builder); + Throw.IfNull(tools); + + foreach (var tool in tools) + { + builder.WithAITool(tool); + } + + return builder; + } } diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting/Local/LocalAgentCatalog.cs b/dotnet/src/Microsoft.Agents.AI.Hosting/Local/LocalAgentCatalog.cs deleted file mode 100644 index 0b44ad60cb..0000000000 --- a/dotnet/src/Microsoft.Agents.AI.Hosting/Local/LocalAgentCatalog.cs +++ /dev/null @@ -1,37 +0,0 @@ -// Copyright (c) Microsoft. All rights reserved. - -using System; -using System.Collections.Generic; -using System.Runtime.CompilerServices; -using System.Threading; -using System.Threading.Tasks; -using Microsoft.Extensions.DependencyInjection; - -namespace Microsoft.Agents.AI.Hosting.Local; - -// Implementation of an AgentCatalog which enumerates agents registered in the local service provider. -internal sealed class LocalAgentCatalog : AgentCatalog -{ - public readonly HashSet _registeredAgents; - private readonly IServiceProvider _serviceProvider; - - public LocalAgentCatalog(LocalAgentRegistry agentHostBuilder, IServiceProvider serviceProvider) - { - this._registeredAgents = [.. agentHostBuilder.AgentNames]; - this._serviceProvider = serviceProvider; - } - - public override async IAsyncEnumerable GetAgentsAsync([EnumeratorCancellation] CancellationToken cancellationToken = default) - { - await Task.CompletedTask.ConfigureAwait(false); - - foreach (var name in this._registeredAgents) - { - var agent = this._serviceProvider.GetKeyedService(name); - if (agent is not null) - { - yield return agent; - } - } - } -} diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting/Local/LocalAgentRegistry.cs b/dotnet/src/Microsoft.Agents.AI.Hosting/Local/LocalAgentRegistry.cs deleted file mode 100644 index df3db8f554..0000000000 --- a/dotnet/src/Microsoft.Agents.AI.Hosting/Local/LocalAgentRegistry.cs +++ /dev/null @@ -1,10 +0,0 @@ -// Copyright (c) Microsoft. All rights reserved. - -using System.Collections.Generic; - -namespace Microsoft.Agents.AI.Hosting.Local; - -internal sealed class LocalAgentRegistry -{ - public HashSet AgentNames { get; } = []; -} diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting/Local/LocalAgentToolRegistry.cs b/dotnet/src/Microsoft.Agents.AI.Hosting/Local/LocalAgentToolRegistry.cs new file mode 100644 index 0000000000..8c87803db3 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Hosting/Local/LocalAgentToolRegistry.cs @@ -0,0 +1,27 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Collections.Generic; +using Microsoft.Extensions.AI; + +namespace Microsoft.Agents.AI.Hosting.Local; + +internal sealed class LocalAgentToolRegistry +{ + private readonly Dictionary> _toolsByAgentName = []; + + public void AddTool(string agentName, AITool tool) + { + if (!this._toolsByAgentName.TryGetValue(agentName, out var tools)) + { + tools = []; + this._toolsByAgentName[agentName] = tools; + } + + tools.Add(tool); + } + + public IList GetTools(string agentName) + { + return this._toolsByAgentName.TryGetValue(agentName, out var tools) ? tools : []; + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting/Local/LocalWorkflowCatalog.cs b/dotnet/src/Microsoft.Agents.AI.Hosting/Local/LocalWorkflowCatalog.cs deleted file mode 100644 index 572b41830e..0000000000 --- a/dotnet/src/Microsoft.Agents.AI.Hosting/Local/LocalWorkflowCatalog.cs +++ /dev/null @@ -1,37 +0,0 @@ -// Copyright (c) Microsoft. All rights reserved. - -using System; -using System.Collections.Generic; -using System.Runtime.CompilerServices; -using System.Threading; -using System.Threading.Tasks; -using Microsoft.Agents.AI.Workflows; -using Microsoft.Extensions.DependencyInjection; - -namespace Microsoft.Agents.AI.Hosting.Local; - -internal sealed class LocalWorkflowCatalog : WorkflowCatalog -{ - public readonly HashSet _registeredWorkflows; - private readonly IServiceProvider _serviceProvider; - - public LocalWorkflowCatalog(LocalWorkflowRegistry workflowRegistry, IServiceProvider serviceProvider) - { - this._registeredWorkflows = [.. workflowRegistry.WorkflowNames]; - this._serviceProvider = serviceProvider; - } - - public override async IAsyncEnumerable GetWorkflowsAsync([EnumeratorCancellation] CancellationToken cancellationToken = default) - { - await Task.CompletedTask.ConfigureAwait(false); - - foreach (var name in this._registeredWorkflows) - { - var workflow = this._serviceProvider.GetKeyedService(name); - if (workflow is not null) - { - yield return workflow; - } - } - } -} diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting/Local/LocalWorkflowRegistry.cs b/dotnet/src/Microsoft.Agents.AI.Hosting/Local/LocalWorkflowRegistry.cs deleted file mode 100644 index 803c24660f..0000000000 --- a/dotnet/src/Microsoft.Agents.AI.Hosting/Local/LocalWorkflowRegistry.cs +++ /dev/null @@ -1,10 +0,0 @@ -// Copyright (c) Microsoft. All rights reserved. - -using System.Collections.Generic; - -namespace Microsoft.Agents.AI.Hosting.Local; - -internal sealed class LocalWorkflowRegistry -{ - public HashSet WorkflowNames { get; } = []; -} diff --git a/dotnet/src/Microsoft.Agents.AI.Hosting/Microsoft.Agents.AI.Hosting.csproj b/dotnet/src/Microsoft.Agents.AI.Hosting/Microsoft.Agents.AI.Hosting.csproj index 86f709877d..70c690bfdf 100644 --- a/dotnet/src/Microsoft.Agents.AI.Hosting/Microsoft.Agents.AI.Hosting.csproj +++ b/dotnet/src/Microsoft.Agents.AI.Hosting/Microsoft.Agents.AI.Hosting.csproj @@ -1,8 +1,6 @@ - $(ProjectsTargetFrameworks) - $(ProjectsDebugTargetFrameworks) preview diff --git a/dotnet/src/Microsoft.Agents.AI.Mem0/Mem0Client.cs b/dotnet/src/Microsoft.Agents.AI.Mem0/Mem0Client.cs index ad8120c402..39c4db8c96 100644 --- a/dotnet/src/Microsoft.Agents.AI.Mem0/Mem0Client.cs +++ b/dotnet/src/Microsoft.Agents.AI.Mem0/Mem0Client.cs @@ -71,7 +71,7 @@ public async Task> SearchAsync(string? applicationId, string var response = await responseMessage.Content.ReadAsStringAsync().ConfigureAwait(false); #endif var searchResponseItems = JsonSerializer.Deserialize(response, Mem0SourceGenerationContext.Default.SearchResponseItemArray); - return searchResponseItems?.Select(item => item.Memory) ?? Array.Empty(); + return searchResponseItems?.Select(item => item.Memory) ?? []; } /// @@ -94,14 +94,14 @@ public async Task CreateMemoryAsync(string? applicationId, string? agentId, stri AgentId = agentId, RunId = threadId, UserId = userId, - Messages = new[] - { + Messages = + [ new CreateMemoryMessage { Content = messageContent, Role = messageRole.ToLowerInvariant() } - } + ] }; #pragma warning restore CA1308 @@ -133,7 +133,7 @@ internal sealed class CreateMemoryRequest [JsonPropertyName("agent_id")] public string? AgentId { get; set; } [JsonPropertyName("run_id")] public string? RunId { get; set; } [JsonPropertyName("user_id")] public string? UserId { get; set; } - [JsonPropertyName("messages")] public CreateMemoryMessage[] Messages { get; set; } = Array.Empty(); + [JsonPropertyName("messages")] public CreateMemoryMessage[] Messages { get; set; } = []; } internal sealed class CreateMemoryMessage diff --git a/dotnet/src/Microsoft.Agents.AI.Mem0/Mem0Provider.cs b/dotnet/src/Microsoft.Agents.AI.Mem0/Mem0Provider.cs index 4aae5de59b..0e9b4288b1 100644 --- a/dotnet/src/Microsoft.Agents.AI.Mem0/Mem0Provider.cs +++ b/dotnet/src/Microsoft.Agents.AI.Mem0/Mem0Provider.cs @@ -28,6 +28,7 @@ public sealed class Mem0Provider : AIContextProvider private const string DefaultContextPrompt = "## Memories\nConsider the following memories when answering user questions:"; private readonly string _contextPrompt; + private readonly bool _enableSensitiveTelemetryData; private readonly Mem0Client _client; private readonly ILogger? _logger; @@ -64,6 +65,7 @@ public Mem0Provider(HttpClient httpClient, Mem0ProviderScope storageScope, Mem0P this._client = new Mem0Client(httpClient); this._contextPrompt = options?.ContextPrompt ?? DefaultContextPrompt; + this._enableSensitiveTelemetryData = options?.EnableSensitiveTelemetryData ?? false; this._storageScope = new Mem0ProviderScope(Throw.IfNull(storageScope)); this._searchScope = searchScope ?? storageScope; @@ -114,6 +116,7 @@ public Mem0Provider(HttpClient httpClient, JsonElement serializedState, JsonSeri this._client = new Mem0Client(httpClient); this._contextPrompt = options?.ContextPrompt ?? DefaultContextPrompt; + this._enableSensitiveTelemetryData = options?.EnableSensitiveTelemetryData ?? false; var jso = jsonSerializerOptions ?? Mem0JsonUtilities.DefaultOptions; var state = serializedState.Deserialize(jso.GetTypeInfo(typeof(Mem0State))) as Mem0State; @@ -150,25 +153,26 @@ public override async ValueTask InvokingAsync(InvokingContext context ? null : $"{this._contextPrompt}\n{string.Join(Environment.NewLine, memories)}"; - if (this._logger is not null) + if (this._logger?.IsEnabled(LogLevel.Information) is true) { this._logger.LogInformation( - "Mem0AIContextProvider: Retrieved {Count} memories. ApplicationId: '{ApplicationId}', AgentId: '{AgentId}', ThreadId: '{ThreadId}', UserId: '{UserId}'", + "Mem0AIContextProvider: Retrieved {Count} memories. ApplicationId: '{ApplicationId}', AgentId: '{AgentId}', ThreadId: '{ThreadId}', UserId: '{UserId}'.", memories.Count, this._searchScope.ApplicationId, this._searchScope.AgentId, this._searchScope.ThreadId, - this._searchScope.UserId); - if (outputMessageText is not null) + this.SanitizeLogData(this._searchScope.UserId)); + + if (outputMessageText is not null && this._logger.IsEnabled(LogLevel.Trace)) { this._logger.LogTrace( - "Mem0AIContextProvider: Search Results\nInput:{Input}\nOutput:{MessageText}\nApplicationId: '{ApplicationId}', AgentId: '{AgentId}', ThreadId: '{ThreadId}', UserId: '{UserId}'", - queryText, - outputMessageText, + "Mem0AIContextProvider: Search Results\nInput:{Input}\nOutput:{MessageText}\nApplicationId: '{ApplicationId}', AgentId: '{AgentId}', ThreadId: '{ThreadId}', UserId: '{UserId}'.", + this.SanitizeLogData(queryText), + this.SanitizeLogData(outputMessageText), this._searchScope.ApplicationId, this._searchScope.AgentId, this._searchScope.ThreadId, - this._searchScope.UserId); + this.SanitizeLogData(this._searchScope.UserId)); } } @@ -183,13 +187,16 @@ public override async ValueTask InvokingAsync(InvokingContext context } catch (Exception ex) { - this._logger?.LogError( - ex, - "Mem0AIContextProvider: Failed to search Mem0 for memories due to error. ApplicationId: '{ApplicationId}', AgentId: '{AgentId}', ThreadId: '{ThreadId}', UserId: '{UserId}'", - this._searchScope.ApplicationId, - this._searchScope.AgentId, - this._searchScope.ThreadId, - this._searchScope.UserId); + if (this._logger?.IsEnabled(LogLevel.Error) is true) + { + this._logger.LogError( + ex, + "Mem0AIContextProvider: Failed to search Mem0 for memories due to error. ApplicationId: '{ApplicationId}', AgentId: '{AgentId}', ThreadId: '{ThreadId}', UserId: '{UserId}'.", + this._searchScope.ApplicationId, + this._searchScope.AgentId, + this._searchScope.ThreadId, + this.SanitizeLogData(this._searchScope.UserId)); + } return new AIContext(); } } @@ -209,13 +216,16 @@ public override async ValueTask InvokedAsync(InvokedContext context, Cancellatio } catch (Exception ex) { - this._logger?.LogError( - ex, - "Mem0AIContextProvider: Failed to send messages to Mem0 due to error. ApplicationId: '{ApplicationId}', AgentId: '{AgentId}', ThreadId: '{ThreadId}', UserId: '{UserId}'", - this._storageScope.ApplicationId, - this._storageScope.AgentId, - this._storageScope.ThreadId, - this._storageScope.UserId); + if (this._logger?.IsEnabled(LogLevel.Error) is true) + { + this._logger.LogError( + ex, + "Mem0AIContextProvider: Failed to send messages to Mem0 due to error. ApplicationId: '{ApplicationId}', AgentId: '{AgentId}', ThreadId: '{ThreadId}', UserId: '{UserId}'.", + this._storageScope.ApplicationId, + this._storageScope.AgentId, + this._storageScope.ThreadId, + this.SanitizeLogData(this._storageScope.UserId)); + } } } @@ -282,4 +292,6 @@ public Mem0State(Mem0ProviderScope storageScope, Mem0ProviderScope searchScope) public Mem0ProviderScope StorageScope { get; set; } public Mem0ProviderScope SearchScope { get; set; } } + + private string? SanitizeLogData(string? data) => this._enableSensitiveTelemetryData ? data : ""; } diff --git a/dotnet/src/Microsoft.Agents.AI.Mem0/Mem0ProviderOptions.cs b/dotnet/src/Microsoft.Agents.AI.Mem0/Mem0ProviderOptions.cs index 34b0392bec..f2d3d89e16 100644 --- a/dotnet/src/Microsoft.Agents.AI.Mem0/Mem0ProviderOptions.cs +++ b/dotnet/src/Microsoft.Agents.AI.Mem0/Mem0ProviderOptions.cs @@ -12,4 +12,10 @@ public sealed class Mem0ProviderOptions /// /// Defaults to "## Memories\nConsider the following memories when answering user questions:". public string? ContextPrompt { get; set; } + + /// + /// Gets or sets a value indicating whether sensitive data such as user ids and user messages may appear in logs. + /// + /// Defaults to . + public bool EnableSensitiveTelemetryData { get; set; } } diff --git a/dotnet/src/Microsoft.Agents.AI.Mem0/Microsoft.Agents.AI.Mem0.csproj b/dotnet/src/Microsoft.Agents.AI.Mem0/Microsoft.Agents.AI.Mem0.csproj index e78e93c955..19a5019843 100644 --- a/dotnet/src/Microsoft.Agents.AI.Mem0/Microsoft.Agents.AI.Mem0.csproj +++ b/dotnet/src/Microsoft.Agents.AI.Mem0/Microsoft.Agents.AI.Mem0.csproj @@ -1,8 +1,6 @@  - $(ProjectsTargetFrameworks) - $(ProjectsDebugTargetFrameworks) preview diff --git a/dotnet/src/Microsoft.Agents.AI.OpenAI/ChatClient/AsyncStreamingUpdateCollectionResult.cs b/dotnet/src/Microsoft.Agents.AI.OpenAI/ChatClient/AsyncStreamingChatCompletionUpdateCollectionResult.cs similarity index 78% rename from dotnet/src/Microsoft.Agents.AI.OpenAI/ChatClient/AsyncStreamingUpdateCollectionResult.cs rename to dotnet/src/Microsoft.Agents.AI.OpenAI/ChatClient/AsyncStreamingChatCompletionUpdateCollectionResult.cs index a118c6c1de..17c9c2d95a 100644 --- a/dotnet/src/Microsoft.Agents.AI.OpenAI/ChatClient/AsyncStreamingUpdateCollectionResult.cs +++ b/dotnet/src/Microsoft.Agents.AI.OpenAI/ChatClient/AsyncStreamingChatCompletionUpdateCollectionResult.cs @@ -5,11 +5,11 @@ namespace Microsoft.Agents.AI.OpenAI; -internal sealed class AsyncStreamingUpdateCollectionResult : AsyncCollectionResult +internal sealed class AsyncStreamingChatCompletionUpdateCollectionResult : AsyncCollectionResult { private readonly IAsyncEnumerable _updates; - internal AsyncStreamingUpdateCollectionResult(IAsyncEnumerable updates) + internal AsyncStreamingChatCompletionUpdateCollectionResult(IAsyncEnumerable updates) { this._updates = updates; } diff --git a/dotnet/src/Microsoft.Agents.AI.OpenAI/ChatClient/AsyncStreamingResponseUpdateCollectionResult.cs b/dotnet/src/Microsoft.Agents.AI.OpenAI/ChatClient/AsyncStreamingResponseUpdateCollectionResult.cs new file mode 100644 index 0000000000..c67f4d1462 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.OpenAI/ChatClient/AsyncStreamingResponseUpdateCollectionResult.cs @@ -0,0 +1,48 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.ClientModel; +using OpenAI.Responses; + +namespace Microsoft.Agents.AI.OpenAI; + +internal sealed class AsyncStreamingResponseUpdateCollectionResult : AsyncCollectionResult +{ + private readonly IAsyncEnumerable _updates; + + internal AsyncStreamingResponseUpdateCollectionResult(IAsyncEnumerable updates) + { + this._updates = updates; + } + + public override ContinuationToken? GetContinuationToken(ClientResult page) => null; + + public override async IAsyncEnumerable GetRawPagesAsync() + { + yield return ClientResult.FromValue(this._updates, new StreamingUpdatePipelineResponse(this._updates)); + } + + protected async override IAsyncEnumerable GetValuesFromPageAsync(ClientResult page) + { + var updates = ((ClientResult>)page).Value; + + await foreach (var update in updates.ConfigureAwait(false)) + { + switch (update.RawRepresentation) + { + case StreamingResponseUpdate rawUpdate: + yield return rawUpdate; + break; + + case Extensions.AI.ChatResponseUpdate { RawRepresentation: StreamingResponseUpdate rawUpdate }: + yield return rawUpdate; + break; + + default: + // TODO: The OpenAI library does not currently expose model factory methods for creating + // StreamingResponseUpdates. We are thus unable to manufacture such instances when there isn't + // already one in the update and instead skip them. + break; + } + } + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.OpenAI/Extensions/AIAgentWithOpenAIExtensions.cs b/dotnet/src/Microsoft.Agents.AI.OpenAI/Extensions/AIAgentWithOpenAIExtensions.cs index 32bb50080d..4abc6915a6 100644 --- a/dotnet/src/Microsoft.Agents.AI.OpenAI/Extensions/AIAgentWithOpenAIExtensions.cs +++ b/dotnet/src/Microsoft.Agents.AI.OpenAI/Extensions/AIAgentWithOpenAIExtensions.cs @@ -1,12 +1,12 @@ // Copyright (c) Microsoft. All rights reserved. using System.ClientModel; -using Microsoft.Agents.AI; using Microsoft.Agents.AI.OpenAI; using Microsoft.Shared.Diagnostics; using OpenAI.Chat; +using OpenAI.Responses; -namespace OpenAI; +namespace Microsoft.Agents.AI; /// /// Provides extension methods for to simplify interaction with OpenAI chat messages @@ -69,6 +69,60 @@ public static AsyncCollectionResult RunStreamingA IAsyncEnumerable response = agent.RunStreamingAsync([.. messages.AsChatMessages()], thread, options, cancellationToken); - return new AsyncStreamingUpdateCollectionResult(response); + return new AsyncStreamingChatCompletionUpdateCollectionResult(response); + } + + /// + /// Runs the AI agent with a collection of OpenAI response items and returns the response as a native OpenAI . + /// + /// The AI agent to run. + /// The collection of OpenAI response items to send to the agent. + /// The conversation thread to continue with this invocation. If not provided, creates a new thread. The thread will be mutated with the provided messages and agent response. + /// Optional parameters for agent invocation. + /// The to monitor for cancellation requests. The default is . + /// A representing the asynchronous operation that returns a native OpenAI response. + /// Thrown when or is . + /// Thrown when the agent's response cannot be converted to an , typically when the underlying representation is not an OpenAI response. + /// Thrown when any message in has a type that is not supported by the message conversion method. + /// + /// This method converts the OpenAI response items to the Microsoft Extensions AI format using the appropriate conversion method, + /// runs the agent with the converted message collection, and then extracts the native OpenAI from the response using . + /// + public static async Task RunAsync(this AIAgent agent, IEnumerable messages, AgentThread? thread = null, AgentRunOptions? options = null, CancellationToken cancellationToken = default) + { + Throw.IfNull(agent); + Throw.IfNull(messages); + + var response = await agent.RunAsync(messages.AsChatMessages(), thread, options, cancellationToken).ConfigureAwait(false); + + return response.AsOpenAIResponse(); + } + + /// + /// Runs the AI agent in streaming mode with a collection of OpenAI response items and returns the response as a collection of native OpenAI . + /// + /// The AI agent to run. + /// The collection of OpenAI response items to send to the agent. + /// The conversation thread to continue with this invocation. If not provided, creates a new thread. The thread will be mutated with the provided messages and agent response updates. + /// Optional parameters for agent invocation. + /// The to monitor for cancellation requests. The default is . + /// An representing the asynchronous enumerable that yields native OpenAI instances as they are streamed. + /// Thrown when or is . + /// Thrown when the agent's response cannot be converted to instances, typically when the underlying representation is not an OpenAI response. + /// Thrown when any message in has a type that is not supported by the message conversion method. + /// + /// This method converts the OpenAI response items to the Microsoft Extensions AI format using the appropriate conversion method, + /// runs the agent in streaming mode, and then yields native OpenAI instances as they are produced. + /// The method attempts to extract from the underlying response representation. If a raw update is not available, + /// it is skipped because the OpenAI library does not currently expose model factory methods for creating such instances. + /// + public static AsyncCollectionResult RunStreamingAsync(this AIAgent agent, IEnumerable messages, AgentThread? thread = null, AgentRunOptions? options = null, CancellationToken cancellationToken = default) + { + Throw.IfNull(agent); + Throw.IfNull(messages); + + IAsyncEnumerable response = agent.RunStreamingAsync([.. messages.AsChatMessages()], thread, options, cancellationToken); + + return new AsyncStreamingResponseUpdateCollectionResult(response); } } diff --git a/dotnet/src/Microsoft.Agents.AI.OpenAI/Extensions/AgentRunResponseExtensions.cs b/dotnet/src/Microsoft.Agents.AI.OpenAI/Extensions/AgentRunResponseExtensions.cs index 1660192fad..9a164d862b 100644 --- a/dotnet/src/Microsoft.Agents.AI.OpenAI/Extensions/AgentRunResponseExtensions.cs +++ b/dotnet/src/Microsoft.Agents.AI.OpenAI/Extensions/AgentRunResponseExtensions.cs @@ -5,7 +5,7 @@ using OpenAI.Chat; using OpenAI.Responses; -namespace OpenAI; +namespace Microsoft.Agents.AI; /// /// Provides extension methods for and instances to diff --git a/dotnet/src/Microsoft.Agents.AI.OpenAI/Extensions/OpenAIAssistantClientExtensions.cs b/dotnet/src/Microsoft.Agents.AI.OpenAI/Extensions/OpenAIAssistantClientExtensions.cs index 71f9b5436b..881266fe8b 100644 --- a/dotnet/src/Microsoft.Agents.AI.OpenAI/Extensions/OpenAIAssistantClientExtensions.cs +++ b/dotnet/src/Microsoft.Agents.AI.OpenAI/Extensions/OpenAIAssistantClientExtensions.cs @@ -5,9 +5,8 @@ using Microsoft.Extensions.AI; using Microsoft.Extensions.Logging; using Microsoft.Shared.Diagnostics; -using OpenAI.Assistants; -namespace OpenAI; +namespace OpenAI.Assistants; /// /// Provides extension methods for OpenAI @@ -28,19 +27,22 @@ public static class OpenAIAssistantClientExtensions /// The client result containing the assistant. /// Optional chat options. /// Provides a way to customize the creation of the underlying used by the agent. + /// An optional to use for resolving services required by the instances being invoked. /// A instance that can be used to perform operations on the assistant. + [Obsolete("The Assistants API has been deprecated. Please use the Responses API instead.")] public static ChatClientAgent GetAIAgent( this AssistantClient assistantClient, ClientResult assistantClientResult, ChatOptions? chatOptions = null, - Func? clientFactory = null) + Func? clientFactory = null, + IServiceProvider? services = null) { if (assistantClientResult is null) { throw new ArgumentNullException(nameof(assistantClientResult)); } - return assistantClient.GetAIAgent(assistantClientResult.Value, chatOptions, clientFactory); + return assistantClient.GetAIAgent(assistantClientResult.Value, chatOptions, clientFactory, services); } /// @@ -50,12 +52,15 @@ public static ChatClientAgent GetAIAgent( /// The assistant metadata. /// Optional chat options. /// Provides a way to customize the creation of the underlying used by the agent. + /// An optional to use for resolving services required by the instances being invoked. /// A instance that can be used to perform operations on the assistant. + [Obsolete("The Assistants API has been deprecated. Please use the Responses API instead.")] public static ChatClientAgent GetAIAgent( this AssistantClient assistantClient, Assistant assistantMetadata, ChatOptions? chatOptions = null, - Func? clientFactory = null) + Func? clientFactory = null, + IServiceProvider? services = null) { if (assistantMetadata is null) { @@ -73,14 +78,19 @@ public static ChatClientAgent GetAIAgent( chatClient = clientFactory(chatClient); } + if (!string.IsNullOrWhiteSpace(assistantMetadata.Instructions) && chatOptions?.Instructions is null) + { + chatOptions ??= new ChatOptions(); + chatOptions.Instructions = assistantMetadata.Instructions; + } + return new ChatClientAgent(chatClient, options: new() { Id = assistantMetadata.Id, Name = assistantMetadata.Name, Description = assistantMetadata.Description, - Instructions = assistantMetadata.Instructions, ChatOptions = chatOptions - }); + }, services: services); } /// @@ -90,13 +100,16 @@ public static ChatClientAgent GetAIAgent( /// The ID of the server side agent to create a for. /// Options that should apply to all runs of the agent. /// Provides a way to customize the creation of the underlying used by the agent. + /// An optional to use for resolving services required by the instances being invoked. /// The to monitor for cancellation requests. The default is . /// A instance that can be used to perform operations on the assistant agent. + [Obsolete("The Assistants API has been deprecated. Please use the Responses API instead.")] public static ChatClientAgent GetAIAgent( this AssistantClient assistantClient, string agentId, ChatOptions? chatOptions = null, Func? clientFactory = null, + IServiceProvider? services = null, CancellationToken cancellationToken = default) { if (assistantClient is null) @@ -110,7 +123,7 @@ public static ChatClientAgent GetAIAgent( } var assistant = assistantClient.GetAssistant(agentId, cancellationToken); - return assistantClient.GetAIAgent(assistant, chatOptions, clientFactory); + return assistantClient.GetAIAgent(assistant, chatOptions, clientFactory, services); } /// @@ -120,13 +133,16 @@ public static ChatClientAgent GetAIAgent( /// The ID of the server side agent to create a for. /// Options that should apply to all runs of the agent. /// Provides a way to customize the creation of the underlying used by the agent. + /// An optional to use for resolving services required by the instances being invoked. /// The to monitor for cancellation requests. The default is . /// A instance that can be used to perform operations on the assistant agent. + [Obsolete("The Assistants API has been deprecated. Please use the Responses API instead.")] public static async Task GetAIAgentAsync( this AssistantClient assistantClient, string agentId, ChatOptions? chatOptions = null, Func? clientFactory = null, + IServiceProvider? services = null, CancellationToken cancellationToken = default) { if (assistantClient is null) @@ -140,7 +156,7 @@ public static async Task GetAIAgentAsync( } var assistantResponse = await assistantClient.GetAssistantAsync(agentId, cancellationToken).ConfigureAwait(false); - return assistantClient.GetAIAgent(assistantResponse, chatOptions, clientFactory); + return assistantClient.GetAIAgent(assistantResponse, chatOptions, clientFactory, services); } /// @@ -150,20 +166,23 @@ public static async Task GetAIAgentAsync( /// The client result containing the assistant. /// Full set of options to configure the agent. /// Provides a way to customize the creation of the underlying used by the agent. + /// An optional to use for resolving services required by the instances being invoked. /// A instance that can be used to perform operations on the assistant. /// or is . + [Obsolete("The Assistants API has been deprecated. Please use the Responses API instead.")] public static ChatClientAgent GetAIAgent( this AssistantClient assistantClient, ClientResult assistantClientResult, ChatClientAgentOptions options, - Func? clientFactory = null) + Func? clientFactory = null, + IServiceProvider? services = null) { if (assistantClientResult is null) { throw new ArgumentNullException(nameof(assistantClientResult)); } - return assistantClient.GetAIAgent(assistantClientResult.Value, options, clientFactory); + return assistantClient.GetAIAgent(assistantClientResult.Value, options, clientFactory, services); } /// @@ -173,13 +192,16 @@ public static ChatClientAgent GetAIAgent( /// The assistant metadata. /// Full set of options to configure the agent. /// Provides a way to customize the creation of the underlying used by the agent. + /// An optional to use for resolving services required by the instances being invoked. /// A instance that can be used to perform operations on the assistant. /// or is . + [Obsolete("The Assistants API has been deprecated. Please use the Responses API instead.")] public static ChatClientAgent GetAIAgent( this AssistantClient assistantClient, Assistant assistantMetadata, ChatClientAgentOptions options, - Func? clientFactory = null) + Func? clientFactory = null, + IServiceProvider? services = null) { if (assistantMetadata is null) { @@ -203,19 +225,24 @@ public static ChatClientAgent GetAIAgent( chatClient = clientFactory(chatClient); } + if (string.IsNullOrWhiteSpace(options.ChatOptions?.Instructions) && !string.IsNullOrWhiteSpace(assistantMetadata.Instructions)) + { + options.ChatOptions ??= new ChatOptions(); + options.ChatOptions.Instructions = assistantMetadata.Instructions; + } + var mergedOptions = new ChatClientAgentOptions() { Id = assistantMetadata.Id, Name = options.Name ?? assistantMetadata.Name, Description = options.Description ?? assistantMetadata.Description, - Instructions = options.Instructions ?? assistantMetadata.Instructions, ChatOptions = options.ChatOptions, AIContextProviderFactory = options.AIContextProviderFactory, ChatMessageStoreFactory = options.ChatMessageStoreFactory, UseProvidedChatClientAsIs = options.UseProvidedChatClientAsIs }; - return new ChatClientAgent(chatClient, mergedOptions); + return new ChatClientAgent(chatClient, mergedOptions, services: services); } /// @@ -225,15 +252,18 @@ public static ChatClientAgent GetAIAgent( /// The ID of the server side agent to create a for. /// Full set of options to configure the agent. /// Provides a way to customize the creation of the underlying used by the agent. + /// An optional to use for resolving services required by the instances being invoked. /// The to monitor for cancellation requests. The default is . /// A instance that can be used to perform operations on the assistant agent. /// or is . /// is empty or whitespace. + [Obsolete("The Assistants API has been deprecated. Please use the Responses API instead.")] public static ChatClientAgent GetAIAgent( this AssistantClient assistantClient, string agentId, ChatClientAgentOptions options, Func? clientFactory = null, + IServiceProvider? services = null, CancellationToken cancellationToken = default) { if (assistantClient is null) @@ -252,7 +282,7 @@ public static ChatClientAgent GetAIAgent( } var assistant = assistantClient.GetAssistant(agentId, cancellationToken); - return assistantClient.GetAIAgent(assistant, options, clientFactory); + return assistantClient.GetAIAgent(assistant, options, clientFactory, services); } /// @@ -262,15 +292,18 @@ public static ChatClientAgent GetAIAgent( /// The ID of the server side agent to create a for. /// Full set of options to configure the agent. /// Provides a way to customize the creation of the underlying used by the agent. + /// An optional to use for resolving services required by the instances being invoked. /// The to monitor for cancellation requests. The default is . /// A instance that can be used to perform operations on the assistant agent. /// or is . /// is empty or whitespace. + [Obsolete("The Assistants API has been deprecated. Please use the Responses API instead.")] public static async Task GetAIAgentAsync( this AssistantClient assistantClient, string agentId, ChatClientAgentOptions options, Func? clientFactory = null, + IServiceProvider? services = null, CancellationToken cancellationToken = default) { if (assistantClient is null) @@ -289,7 +322,7 @@ public static async Task GetAIAgentAsync( } var assistantResponse = await assistantClient.GetAssistantAsync(agentId, cancellationToken).ConfigureAwait(false); - return assistantClient.GetAIAgent(assistantResponse, options, clientFactory); + return assistantClient.GetAIAgent(assistantResponse, options, clientFactory, services); } /// @@ -303,9 +336,11 @@ public static async Task GetAIAgentAsync( /// Optional collection of AI tools that the agent can use during conversations. /// Provides a way to customize the creation of the underlying used by the agent. /// Optional logger factory for enabling logging within the agent. + /// An optional to use for resolving services required by the instances being invoked. /// An instance backed by the OpenAI Assistant service. /// Thrown when or is . /// Thrown when is empty or whitespace. + [Obsolete("The Assistants API has been deprecated. Please use the Responses API instead.")] public static ChatClientAgent CreateAIAgent( this AssistantClient client, string model, @@ -314,21 +349,23 @@ public static ChatClientAgent CreateAIAgent( string? description = null, IList? tools = null, Func? clientFactory = null, - ILoggerFactory? loggerFactory = null) => + ILoggerFactory? loggerFactory = null, + IServiceProvider? services = null) => client.CreateAIAgent( model, new ChatClientAgentOptions() { Name = name, Description = description, - Instructions = instructions, - ChatOptions = tools is null ? null : new ChatOptions() + ChatOptions = tools is null && string.IsNullOrWhiteSpace(instructions) ? null : new ChatOptions() { Tools = tools, + Instructions = instructions } }, clientFactory, - loggerFactory); + loggerFactory, + services); /// /// Creates an AI agent from an using the OpenAI Assistant API. @@ -338,15 +375,18 @@ public static ChatClientAgent CreateAIAgent( /// Full set of options to configure the agent. /// Provides a way to customize the creation of the underlying used by the agent. /// Optional logger factory for enabling logging within the agent. + /// An optional to use for resolving services required by the instances being invoked. /// An instance backed by the OpenAI Assistant service. /// Thrown when or or is . /// Thrown when is empty or whitespace. + [Obsolete("The Assistants API has been deprecated. Please use the Responses API instead.")] public static ChatClientAgent CreateAIAgent( this AssistantClient client, string model, ChatClientAgentOptions options, Func? clientFactory = null, - ILoggerFactory? loggerFactory = null) + ILoggerFactory? loggerFactory = null, + IServiceProvider? services = null) { Throw.IfNull(client); Throw.IfNullOrEmpty(model); @@ -356,7 +396,7 @@ public static ChatClientAgent CreateAIAgent( { Name = options.Name, Description = options.Description, - Instructions = options.Instructions, + Instructions = options.ChatOptions?.Instructions, }; // Convert AITools to ToolDefinitions and ToolResources @@ -387,7 +427,7 @@ public static ChatClientAgent CreateAIAgent( options.ChatOptions ??= new ChatOptions(); options.ChatOptions!.Tools = toolDefinitionsAndResources.FunctionToolsAndOtherTools; - return new ChatClientAgent(chatClient, agentOptions, loggerFactory); + return new ChatClientAgent(chatClient, agentOptions, loggerFactory, services); } /// @@ -401,9 +441,12 @@ public static ChatClientAgent CreateAIAgent( /// Optional collection of AI tools that the agent can use during conversations. /// Provides a way to customize the creation of the underlying used by the agent. /// Optional logger factory for enabling logging within the agent. + /// An optional to use for resolving services required by the instances being invoked. + /// The to monitor for cancellation requests. The default is . /// An instance backed by the OpenAI Assistant service. /// Thrown when or is . /// Thrown when is empty or whitespace. + [Obsolete("The Assistants API has been deprecated. Please use the Responses API instead.")] public static async Task CreateAIAgentAsync( this AssistantClient client, string model, @@ -412,20 +455,24 @@ public static async Task CreateAIAgentAsync( string? description = null, IList? tools = null, Func? clientFactory = null, - ILoggerFactory? loggerFactory = null) => + ILoggerFactory? loggerFactory = null, + IServiceProvider? services = null, + CancellationToken cancellationToken = default) => await client.CreateAIAgentAsync(model, new ChatClientAgentOptions() { Name = name, Description = description, - Instructions = instructions, - ChatOptions = tools is null ? null : new ChatOptions() + ChatOptions = tools is null && string.IsNullOrWhiteSpace(instructions) ? null : new ChatOptions() { Tools = tools, + Instructions = instructions, } }, clientFactory, - loggerFactory).ConfigureAwait(false); + loggerFactory, + services, + cancellationToken).ConfigureAwait(false); /// /// Creates an AI agent from an using the OpenAI Assistant API. @@ -435,15 +482,20 @@ await client.CreateAIAgentAsync(model, /// Full set of options to configure the agent. /// Provides a way to customize the creation of the underlying used by the agent. /// Optional logger factory for enabling logging within the agent. + /// An optional to use for resolving services required by the instances being invoked. + /// The to monitor for cancellation requests. The default is . /// An instance backed by the OpenAI Assistant service. /// Thrown when or is . /// Thrown when is empty or whitespace. + [Obsolete("The Assistants API has been deprecated. Please use the Responses API instead.")] public static async Task CreateAIAgentAsync( this AssistantClient client, string model, ChatClientAgentOptions options, Func? clientFactory = null, - ILoggerFactory? loggerFactory = null) + ILoggerFactory? loggerFactory = null, + IServiceProvider? services = null, + CancellationToken cancellationToken = default) { Throw.IfNull(client); Throw.IfNull(model); @@ -453,7 +505,7 @@ public static async Task CreateAIAgentAsync( { Name = options.Name, Description = options.Description, - Instructions = options.Instructions, + Instructions = options.ChatOptions?.Instructions, }; // Convert AITools to ToolDefinitions and ToolResources @@ -468,7 +520,7 @@ public static async Task CreateAIAgentAsync( } // Create the assistant in the assistant service. - var assistantCreateResult = await client.CreateAssistantAsync(model, assistantOptions).ConfigureAwait(false); + var assistantCreateResult = await client.CreateAssistantAsync(model, assistantOptions, cancellationToken).ConfigureAwait(false); var assistantId = assistantCreateResult.Value.Id; // Build the local agent object. @@ -483,7 +535,7 @@ public static async Task CreateAIAgentAsync( options.ChatOptions ??= new ChatOptions(); options.ChatOptions!.Tools = toolDefinitionsAndResources.FunctionToolsAndOtherTools; - return new ChatClientAgent(chatClient, agentOptions, loggerFactory); + return new ChatClientAgent(chatClient, agentOptions, loggerFactory, services); } private static (List? ToolDefinitions, ToolResources? ToolResources, List? FunctionToolsAndOtherTools) ConvertAIToolsToToolDefinitions(IList? tools) @@ -500,7 +552,7 @@ private static (List? ToolDefinitions, ToolResources? ToolResour { case HostedCodeInterpreterTool codeTool: - toolDefinitions ??= new(); + toolDefinitions ??= []; toolDefinitions.Add(new CodeInterpreterToolDefinition()); if (codeTool.Inputs is { Count: > 0 }) @@ -521,7 +573,7 @@ private static (List? ToolDefinitions, ToolResources? ToolResour break; case HostedFileSearchTool fileSearchTool: - toolDefinitions ??= new(); + toolDefinitions ??= []; toolDefinitions.Add(new FileSearchToolDefinition { MaxResults = fileSearchTool.MaximumResultCount, @@ -544,7 +596,7 @@ private static (List? ToolDefinitions, ToolResources? ToolResour break; default: - functionToolsAndOtherTools ??= new(); + functionToolsAndOtherTools ??= []; functionToolsAndOtherTools.Add(tool); break; } diff --git a/dotnet/src/Microsoft.Agents.AI.OpenAI/Extensions/OpenAIChatClientExtensions.cs b/dotnet/src/Microsoft.Agents.AI.OpenAI/Extensions/OpenAIChatClientExtensions.cs index 36114d009c..aa4f38e5f4 100644 --- a/dotnet/src/Microsoft.Agents.AI.OpenAI/Extensions/OpenAIChatClientExtensions.cs +++ b/dotnet/src/Microsoft.Agents.AI.OpenAI/Extensions/OpenAIChatClientExtensions.cs @@ -4,9 +4,8 @@ using Microsoft.Extensions.AI; using Microsoft.Extensions.Logging; using Microsoft.Shared.Diagnostics; -using OpenAI.Chat; -namespace OpenAI; +namespace OpenAI.Chat; /// /// Provides extension methods for @@ -47,9 +46,9 @@ public static ChatClientAgent CreateAIAgent( { Name = name, Description = description, - Instructions = instructions, - ChatOptions = tools is null ? null : new ChatOptions() + ChatOptions = tools is null && string.IsNullOrWhiteSpace(instructions) ? null : new ChatOptions() { + Instructions = instructions, Tools = tools, } }, diff --git a/dotnet/src/Microsoft.Agents.AI.OpenAI/Extensions/OpenAIResponseClientExtensions.cs b/dotnet/src/Microsoft.Agents.AI.OpenAI/Extensions/OpenAIResponseClientExtensions.cs index c9f2743229..0d48147c77 100644 --- a/dotnet/src/Microsoft.Agents.AI.OpenAI/Extensions/OpenAIResponseClientExtensions.cs +++ b/dotnet/src/Microsoft.Agents.AI.OpenAI/Extensions/OpenAIResponseClientExtensions.cs @@ -4,9 +4,8 @@ using Microsoft.Extensions.AI; using Microsoft.Extensions.Logging; using Microsoft.Shared.Diagnostics; -using OpenAI.Responses; -namespace OpenAI; +namespace OpenAI.Responses; /// /// Provides extension methods for @@ -30,6 +29,7 @@ public static class OpenAIResponseClientExtensions /// Optional collection of AI tools that the agent can use during conversations. /// Provides a way to customize the creation of the underlying used by the agent. /// Optional logger factory for enabling logging within the agent. + /// An optional to use for resolving services required by the instances being invoked. /// An instance backed by the OpenAI Response service. /// Thrown when is . public static ChatClientAgent CreateAIAgent( @@ -39,7 +39,8 @@ public static ChatClientAgent CreateAIAgent( string? description = null, IList? tools = null, Func? clientFactory = null, - ILoggerFactory? loggerFactory = null) + ILoggerFactory? loggerFactory = null, + IServiceProvider? services = null) { Throw.IfNull(client); @@ -48,14 +49,15 @@ public static ChatClientAgent CreateAIAgent( { Name = name, Description = description, - Instructions = instructions, - ChatOptions = tools is null ? null : new ChatOptions() + ChatOptions = tools is null && string.IsNullOrWhiteSpace(instructions) ? null : new ChatOptions() { + Instructions = instructions, Tools = tools, } }, clientFactory, - loggerFactory); + loggerFactory, + services); } /// @@ -65,13 +67,15 @@ public static ChatClientAgent CreateAIAgent( /// Full set of options to configure the agent. /// Provides a way to customize the creation of the underlying used by the agent. /// Optional logger factory for enabling logging within the agent. + /// An optional to use for resolving services required by the instances being invoked. /// An instance backed by the OpenAI Response service. /// Thrown when or is . public static ChatClientAgent CreateAIAgent( this OpenAIResponseClient client, ChatClientAgentOptions options, Func? clientFactory = null, - ILoggerFactory? loggerFactory = null) + ILoggerFactory? loggerFactory = null, + IServiceProvider? services = null) { Throw.IfNull(client); Throw.IfNull(options); @@ -83,6 +87,6 @@ public static ChatClientAgent CreateAIAgent( chatClient = clientFactory(chatClient); } - return new ChatClientAgent(chatClient, options, loggerFactory); + return new ChatClientAgent(chatClient, options, loggerFactory, services); } } diff --git a/dotnet/src/Microsoft.Agents.AI.OpenAI/Microsoft.Agents.AI.OpenAI.csproj b/dotnet/src/Microsoft.Agents.AI.OpenAI/Microsoft.Agents.AI.OpenAI.csproj index 3c79bb3071..bfcf6e5263 100644 --- a/dotnet/src/Microsoft.Agents.AI.OpenAI/Microsoft.Agents.AI.OpenAI.csproj +++ b/dotnet/src/Microsoft.Agents.AI.OpenAI/Microsoft.Agents.AI.OpenAI.csproj @@ -1,8 +1,6 @@ - $(ProjectsTargetFrameworks) - $(ProjectsDebugTargetFrameworks) preview $(NoWarn);OPENAI001; enable diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/BackgroundJobRunner.cs b/dotnet/src/Microsoft.Agents.AI.Purview/BackgroundJobRunner.cs new file mode 100644 index 0000000000..efe376718b --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/BackgroundJobRunner.cs @@ -0,0 +1,73 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Threading; +using System.Threading.Channels; +using System.Threading.Tasks; +using Microsoft.Agents.AI.Purview.Models.Jobs; +using Microsoft.Extensions.Logging; + +namespace Microsoft.Agents.AI.Purview; + +/// +/// Service that runs jobs in background threads. +/// +internal sealed class BackgroundJobRunner +{ + private readonly IChannelHandler _channelHandler; + private readonly IPurviewClient _purviewClient; + private readonly ILogger _logger; + + /// + /// Initializes a new instance of the class. + /// + /// The channel handler used to manage job channels. + /// The Purview client used to send requests to Purview. + /// The logger used to log information about background jobs. + /// The settings used to configure Purview client behavior. + public BackgroundJobRunner(IChannelHandler channelHandler, IPurviewClient purviewClient, ILogger logger, PurviewSettings purviewSettings) + { + this._channelHandler = channelHandler; + this._purviewClient = purviewClient; + this._logger = logger; + + for (int i = 0; i < purviewSettings.MaxConcurrentJobConsumers; i++) + { + this._channelHandler.AddRunner(async (Channel channel) => + { + await foreach (BackgroundJobBase job in channel.Reader.ReadAllAsync().ConfigureAwait(false)) + { + try + { + await this.RunJobAsync(job).ConfigureAwait(false); + } + catch (Exception e) when (e is not OperationCanceledException and not SystemException) + { + if (this._logger.IsEnabled(LogLevel.Error)) + { + this._logger.LogError(e, "Error running background job {BackgroundJobError}.", e.Message); + } + } + } + }); + } + } + + /// + /// Runs a job. + /// + /// The job to run. + /// A task representing the job. + private async Task RunJobAsync(BackgroundJobBase job) + { + switch (job) + { + case ProcessContentJob processContentJob: + _ = await this._purviewClient.ProcessContentAsync(processContentJob.Request, CancellationToken.None).ConfigureAwait(false); + break; + case ContentActivityJob contentActivityJob: + _ = await this._purviewClient.SendContentActivitiesAsync(contentActivityJob.Request, CancellationToken.None).ConfigureAwait(false); + break; + } + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/CacheProvider.cs b/dotnet/src/Microsoft.Agents.AI.Purview/CacheProvider.cs new file mode 100644 index 0000000000..472b53c50b --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/CacheProvider.cs @@ -0,0 +1,89 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json; +using System.Text.Json.Serialization.Metadata; +using System.Threading; +using System.Threading.Tasks; +using Microsoft.Agents.AI.Purview.Serialization; +using Microsoft.Extensions.Caching.Distributed; + +namespace Microsoft.Agents.AI.Purview; + +/// +/// Manages caching of values. +/// +internal sealed class CacheProvider : ICacheProvider +{ + private readonly IDistributedCache _cache; + private readonly PurviewSettings _purviewSettings; + + /// + /// Create a new instance of the class. + /// + /// The cache where the data is stored. + /// The purview integration settings. + public CacheProvider(IDistributedCache cache, PurviewSettings purviewSettings) + { + this._cache = cache; + this._purviewSettings = purviewSettings; + } + + /// + /// Get a value from the cache. + /// + /// The type of the key in the cache. Used for serialization. + /// The type of the value in the cache. Used for serialization. + /// The key to look up in the cache. + /// A cancellation token for the async operation. + /// The value in the cache. Null or default if no value is present. + public async Task GetAsync(TKey key, CancellationToken cancellationToken) + { + JsonTypeInfo keyTypeInfo = (JsonTypeInfo)PurviewSerializationUtils.SerializationSettings.GetTypeInfo(typeof(TKey)); + string serializedKey = JsonSerializer.Serialize(key, keyTypeInfo); + byte[]? data = await this._cache.GetAsync(serializedKey, cancellationToken).ConfigureAwait(false); + if (data == null) + { + return default; + } + + JsonTypeInfo valueTypeInfo = (JsonTypeInfo)PurviewSerializationUtils.SerializationSettings.GetTypeInfo(typeof(TValue)); + + return JsonSerializer.Deserialize(data, valueTypeInfo); + } + + /// + /// Set a value in the cache. + /// + /// The type of the key in the cache. Used for serialization. + /// The type of the value in the cache. Used for serialization. + /// The key to identify the cache entry. + /// The value to cache. + /// A cancellation token for the async operation. + /// A task for the async operation. + public Task SetAsync(TKey key, TValue value, CancellationToken cancellationToken) + { + JsonTypeInfo keyTypeInfo = (JsonTypeInfo)PurviewSerializationUtils.SerializationSettings.GetTypeInfo(typeof(TKey)); + string serializedKey = JsonSerializer.Serialize(key, keyTypeInfo); + JsonTypeInfo valueTypeInfo = (JsonTypeInfo)PurviewSerializationUtils.SerializationSettings.GetTypeInfo(typeof(TValue)); + byte[] serializedValue = JsonSerializer.SerializeToUtf8Bytes(value, valueTypeInfo); + + DistributedCacheEntryOptions cacheOptions = new() { AbsoluteExpirationRelativeToNow = this._purviewSettings.CacheTTL }; + + return this._cache.SetAsync(serializedKey, serializedValue, cacheOptions, cancellationToken); + } + + /// + /// Removes a value from the cache. + /// + /// The type of the key. + /// The key to identify the cache entry. + /// The cancellation token for the async operation. + /// A task for the async operation. + public Task RemoveAsync(TKey key, CancellationToken cancellationToken) + { + JsonTypeInfo keyTypeInfo = (JsonTypeInfo)PurviewSerializationUtils.SerializationSettings.GetTypeInfo(typeof(TKey)); + string serializedKey = JsonSerializer.Serialize(key, keyTypeInfo); + + return this._cache.RemoveAsync(serializedKey, cancellationToken); + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/ChannelHandler.cs b/dotnet/src/Microsoft.Agents.AI.Purview/ChannelHandler.cs new file mode 100644 index 0000000000..89b5c864fa --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/ChannelHandler.cs @@ -0,0 +1,96 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Collections.Generic; +using System.Threading.Channels; +using System.Threading.Tasks; +using Microsoft.Agents.AI.Purview.Models.Jobs; +using Microsoft.Extensions.Logging; + +namespace Microsoft.Agents.AI.Purview; + +/// +/// Handler class for background job management. +/// +internal class ChannelHandler : IChannelHandler +{ + private readonly Channel _jobChannel; + private readonly List _channelListeners; + private readonly ILogger _logger; + private readonly PurviewSettings _purviewSettings; + + /// + /// Creates a new instance of JobHandler. + /// + /// The purview integration settings. + /// The logger used for logging job information. + /// The job channel used for queuing and reading background jobs. + public ChannelHandler(PurviewSettings purviewSettings, ILogger logger, Channel jobChannel) + { + this._purviewSettings = purviewSettings; + this._logger = logger; + this._jobChannel = jobChannel; + + this._channelListeners = new List(this._purviewSettings.MaxConcurrentJobConsumers); + } + + /// + public void QueueJob(BackgroundJobBase job) + { + try + { + if (job == null) + { + throw new PurviewJobException("Cannot queue null job."); + } + + if (this._channelListeners.Count == 0) + { + this._logger.LogWarning("No listeners are available to process the job."); + throw new PurviewJobException("No listeners are available to process the job."); + } + + bool canQueue = this._jobChannel.Writer.TryWrite(job); + + if (!canQueue) + { + int jobCount = this._jobChannel.Reader.Count; + this._logger.LogError("Could not queue a job for background processing."); + + if (this._jobChannel.Reader.Completion.IsCompleted) + { + throw new PurviewJobException("Job channel is closed or completed. Cannot queue job."); + } + else if (jobCount >= this._purviewSettings.PendingBackgroundJobLimit) + { + throw new PurviewJobLimitExceededException($"Job queue is full. Current pending jobs: {jobCount}. Maximum number of queued jobs: {this._purviewSettings.PendingBackgroundJobLimit}"); + } + else + { + throw new PurviewJobException("Could not queue job for background processing."); + } + } + } + catch (Exception e) when (this._purviewSettings.IgnoreExceptions) + { + if (this._logger.IsEnabled(LogLevel.Error)) + { + this._logger.LogError(e, "Error queuing job: {ExceptionMessage}", e.Message); + } + } + } + + /// + public void AddRunner(Func, Task> runnerTask) + { + this._channelListeners.Add(Task.Run(async () => await runnerTask(this._jobChannel).ConfigureAwait(false))); + } + + /// + public async Task StopAndWaitForCompletionAsync() + { + this._jobChannel.Writer.Complete(); + await this._jobChannel.Reader.Completion.ConfigureAwait(false); + await Task.WhenAll(this._channelListeners).ConfigureAwait(false); + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Constants.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Constants.cs new file mode 100644 index 0000000000..610f0748bc --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Constants.cs @@ -0,0 +1,29 @@ +// Copyright (c) Microsoft. All rights reserved. + +namespace Microsoft.Agents.AI.Purview; + +/// +/// Shared constants for the Purview service. +/// +internal static class Constants +{ + /// + /// The odata type property name used in requests and responses. + /// + public const string ODataTypePropertyName = "@odata.type"; + + /// + /// The OData Graph namespace used for odata types. + /// + public const string ODataGraphNamespace = "microsoft.graph"; + + /// + /// The name of the property that contains the conversation id. + /// + public const string ConversationId = "conversationId"; + + /// + /// The name of the property that contains the user id. + /// + public const string UserId = "userId"; +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Exceptions/PurviewAuthenticationException.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Exceptions/PurviewAuthenticationException.cs new file mode 100644 index 0000000000..83f80f3eb8 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Exceptions/PurviewAuthenticationException.cs @@ -0,0 +1,27 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; + +namespace Microsoft.Agents.AI.Purview; + +/// +/// Exception for authentication errors related to Purview. +/// +public class PurviewAuthenticationException : PurviewException +{ + /// + public PurviewAuthenticationException(string message) + : base(message) + { + } + + /// + public PurviewAuthenticationException() : base() + { + } + + /// + public PurviewAuthenticationException(string? message, Exception? innerException) : base(message, innerException) + { + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Exceptions/PurviewException.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Exceptions/PurviewException.cs new file mode 100644 index 0000000000..36c859d9b1 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Exceptions/PurviewException.cs @@ -0,0 +1,27 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; + +namespace Microsoft.Agents.AI.Purview; + +/// +/// General base exception type for Purview service errors. +/// +public class PurviewException : Exception +{ + /// + public PurviewException(string message) + : base(message) + { + } + + /// + public PurviewException() : base() + { + } + + /// + public PurviewException(string? message, Exception? innerException) : base(message, innerException) + { + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Exceptions/PurviewJobException.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Exceptions/PurviewJobException.cs new file mode 100644 index 0000000000..1737b70f1f --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Exceptions/PurviewJobException.cs @@ -0,0 +1,27 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; + +namespace Microsoft.Agents.AI.Purview; + +/// +/// Represents errors that occur during the execution of a Purview job. +/// +/// This exception is thrown when a Purview job encounters an error that prevents it from completing successfully. +internal class PurviewJobException : PurviewException +{ + /// + public PurviewJobException(string message) : base(message) + { + } + + /// + public PurviewJobException() : base() + { + } + + /// + public PurviewJobException(string? message, Exception? innerException) : base(message, innerException) + { + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Exceptions/PurviewJobLimitExceededException.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Exceptions/PurviewJobLimitExceededException.cs new file mode 100644 index 0000000000..7560000a55 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Exceptions/PurviewJobLimitExceededException.cs @@ -0,0 +1,27 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; + +namespace Microsoft.Agents.AI.Purview; + +/// +/// Represents an exception that is thrown when the maximum number of concurrent Purview jobs has been exceeded. +/// +/// This exception indicates that the Purview service has reached its limit for concurrent job executions. +internal class PurviewJobLimitExceededException : PurviewJobException +{ + /// + public PurviewJobLimitExceededException(string message) : base(message) + { + } + + /// + public PurviewJobLimitExceededException() : base() + { + } + + /// + public PurviewJobLimitExceededException(string? message, Exception? innerException) : base(message, innerException) + { + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Exceptions/PurviewPaymentRequiredException.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Exceptions/PurviewPaymentRequiredException.cs new file mode 100644 index 0000000000..28a6c70323 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Exceptions/PurviewPaymentRequiredException.cs @@ -0,0 +1,26 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; + +namespace Microsoft.Agents.AI.Purview; + +/// +/// Exception for payment required errors related to Purview. +/// +public class PurviewPaymentRequiredException : PurviewException +{ + /// + public PurviewPaymentRequiredException(string message) : base(message) + { + } + + /// + public PurviewPaymentRequiredException() : base() + { + } + + /// + public PurviewPaymentRequiredException(string? message, Exception? innerException) : base(message, innerException) + { + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Exceptions/PurviewRateLimitException.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Exceptions/PurviewRateLimitException.cs new file mode 100644 index 0000000000..71483886d2 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Exceptions/PurviewRateLimitException.cs @@ -0,0 +1,27 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; + +namespace Microsoft.Agents.AI.Purview; + +/// +/// Exception for rate limit exceeded errors from Purview service. +/// +public class PurviewRateLimitException : PurviewException +{ + /// + public PurviewRateLimitException(string message) + : base(message) + { + } + + /// + public PurviewRateLimitException() : base() + { + } + + /// + public PurviewRateLimitException(string? message, Exception? innerException) : base(message, innerException) + { + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Exceptions/PurviewRequestException.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Exceptions/PurviewRequestException.cs new file mode 100644 index 0000000000..a34fad6ce4 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Exceptions/PurviewRequestException.cs @@ -0,0 +1,40 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Net; + +namespace Microsoft.Agents.AI.Purview; + +/// +/// Exception for general http request errors from Purview. +/// +public class PurviewRequestException : PurviewException +{ + /// + /// HTTP status code returned by the Purview service. + /// + public HttpStatusCode StatusCode { get; } + + /// + public PurviewRequestException(HttpStatusCode statusCode, string endpointName) + : base($"Failed to call {endpointName}. Status code: {statusCode}") + { + this.StatusCode = statusCode; + } + + /// + public PurviewRequestException(string message) + : base(message) + { + } + + /// + public PurviewRequestException() : base() + { + } + + /// + public PurviewRequestException(string? message, Exception? innerException) : base(message, innerException) + { + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/ICacheProvider.cs b/dotnet/src/Microsoft.Agents.AI.Purview/ICacheProvider.cs new file mode 100644 index 0000000000..6d6dad527c --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/ICacheProvider.cs @@ -0,0 +1,42 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Threading; +using System.Threading.Tasks; + +namespace Microsoft.Agents.AI.Purview; + +/// +/// Manages caching of values. +/// +internal interface ICacheProvider +{ + /// + /// Get a value from the cache. + /// + /// The type of the key in the cache. Used for serialization. + /// The type of the value in the cache. Used for serialization. + /// The key to look up in the cache. + /// A cancellation token for the async operation. + /// The value in the cache. Null or default if no value is present. + Task GetAsync(TKey key, CancellationToken cancellationToken); + + /// + /// Set a value in the cache. + /// + /// The type of the key in the cache. Used for serialization. + /// The type of the value in the cache. Used for serialization. + /// The key to identify the cache entry. + /// The value to cache. + /// A cancellation token for the async operation. + /// A task for the async operation. + Task SetAsync(TKey key, TValue value, CancellationToken cancellationToken); + + /// + /// Removes a value from the cache. + /// + /// The type of the key. + /// The key to identify the cache entry. + /// The cancellation token for the async operation. + /// A task for the async operation. + Task RemoveAsync(TKey key, CancellationToken cancellationToken); +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/IChannelHandler.cs b/dotnet/src/Microsoft.Agents.AI.Purview/IChannelHandler.cs new file mode 100644 index 0000000000..d8593abd48 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/IChannelHandler.cs @@ -0,0 +1,32 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Threading.Channels; +using System.Threading.Tasks; +using Microsoft.Agents.AI.Purview.Models.Jobs; + +namespace Microsoft.Agents.AI.Purview; + +/// +/// Interface for a class that controls background job processing. +/// +internal interface IChannelHandler +{ + /// + /// Queue a job for background processing. + /// + /// The job queued for background processing. + void QueueJob(BackgroundJobBase job); + + /// + /// Add a runner to the channel handler. + /// + /// The runner task used to process jobs. + void AddRunner(Func, Task> runnerTask); + + /// + /// Stop the channel and wait for all runners to complete + /// + /// A task representing the job. + Task StopAndWaitForCompletionAsync(); +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/IPurviewClient.cs b/dotnet/src/Microsoft.Agents.AI.Purview/IPurviewClient.cs new file mode 100644 index 0000000000..00de9051ef --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/IPurviewClient.cs @@ -0,0 +1,56 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Threading; +using System.Threading.Tasks; +using Microsoft.Agents.AI.Purview.Models.Common; +using Microsoft.Agents.AI.Purview.Models.Requests; +using Microsoft.Agents.AI.Purview.Models.Responses; + +namespace Microsoft.Agents.AI.Purview; + +/// +/// Defines methods for interacting with the Purview service, including content processing, +/// protection scope management, and activity tracking. +/// +/// This interface provides methods to interact with various Purview APIs. It includes processing content, managing protection +/// scopes, and sending content activity data. Implementations of this interface are expected to handle communication +/// with the Purview service and manage any necessary authentication or error handling. +internal interface IPurviewClient +{ + /// + /// Get user info from auth token. + /// + /// The cancellation token used to cancel async processing. + /// The default tenant id used to retrieve the token and its info. + /// The token info from the token. + /// Throw if the token was invalid or could not be retrieved. + Task GetUserInfoFromTokenAsync(CancellationToken cancellationToken, string? tenantId = default); + + /// + /// Call ProcessContent API. + /// + /// The request containing the content to process. + /// The cancellation token used to cancel async processing. + /// The response from the Purview API. + /// Thrown for validation, auth, and network errors. + Task ProcessContentAsync(ProcessContentRequest request, CancellationToken cancellationToken); + + /// + /// Call user ProtectionScope API. + /// + /// The request containing the protection scopes metadata. + /// The cancellation token used to cancel async processing. + /// The protection scopes that apply to the data sent in the request. + /// Thrown for validation, auth, and network errors. + Task GetProtectionScopesAsync(ProtectionScopesRequest request, CancellationToken cancellationToken); + + /// + /// Call contentActivities API. + /// + /// The request containing the content metadata. Used to generate interaction records. + /// The cancellation token used to cancel async processing. + /// The response from the Purview API. + /// Thrown for validation, auth, and network errors. + Task SendContentActivitiesAsync(ContentActivitiesRequest request, CancellationToken cancellationToken); +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/IScopedContentProcessor.cs b/dotnet/src/Microsoft.Agents.AI.Purview/IScopedContentProcessor.cs new file mode 100644 index 0000000000..059e7c4d2d --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/IScopedContentProcessor.cs @@ -0,0 +1,28 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Collections.Generic; +using System.Threading; +using System.Threading.Tasks; +using Microsoft.Agents.AI.Purview.Models.Common; +using Microsoft.Extensions.AI; + +namespace Microsoft.Agents.AI.Purview; + +/// +/// Orchestrates the processing of scoped content by combining protection scope, process content, and content activities operations. +/// +internal interface IScopedContentProcessor +{ + /// + /// Process a list of messages. + /// The list of messages should be a prompt or response. + /// + /// A list of objects sent to the agent or received from the agent.. + /// The thread where the messages were sent. + /// An activity to indicate prompt or response. + /// Purview settings containing tenant id, app name, etc. + /// The user who sent the prompt or is receiving the response. + /// Cancellation token. + /// A bool indicating if the request should be blocked and the user id of the user who made the request. + Task<(bool shouldBlock, string? userId)> ProcessMessagesAsync(IEnumerable messages, string? threadId, Activity activity, PurviewSettings purviewSettings, string? userId, CancellationToken cancellationToken); +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Microsoft.Agents.AI.Purview.csproj b/dotnet/src/Microsoft.Agents.AI.Purview/Microsoft.Agents.AI.Purview.csproj new file mode 100644 index 0000000000..75c19ad7c9 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Microsoft.Agents.AI.Purview.csproj @@ -0,0 +1,41 @@ + + + + alpha + + + + true + true + true + + + + + + + + + + + + + + + + + + Microsoft.Agents.AI.Purview + Tools to connect generative AI apps to Microsoft Purview. + + + + + + + + + $(NoWarn);CA1812 + + + \ No newline at end of file diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/AIAgentInfo.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/AIAgentInfo.cs new file mode 100644 index 0000000000..15c1fbab00 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/AIAgentInfo.cs @@ -0,0 +1,29 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +namespace Microsoft.Agents.AI.Purview.Models.Common; + +/// +/// Info about an AI agent associated with the content. +/// +internal sealed class AIAgentInfo +{ + /// + /// Gets or sets agent id. + /// + [JsonPropertyName("identifier")] + public string? Identifier { get; set; } + + /// + /// Gets or sets agent name. + /// + [JsonPropertyName("name")] + public string? Name { get; set; } + + /// + /// Gets or sets agent version. + /// + [JsonPropertyName("version")] + public string? Version { get; set; } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/AIInteractionPlugin.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/AIInteractionPlugin.cs new file mode 100644 index 0000000000..d9b56f3911 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/AIInteractionPlugin.cs @@ -0,0 +1,29 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +namespace Microsoft.Agents.AI.Purview.Models.Common; + +/// +/// Represents a plugin used in an AI interaction within the Purview SDK. +/// +internal sealed class AIInteractionPlugin +{ + /// + /// Gets or sets Plugin id. + /// + [JsonPropertyName("identifier")] + public string? Identifier { get; set; } + + /// + /// Gets or sets Plugin Name. + /// + [JsonPropertyName("name")] + public string? Name { get; set; } + + /// + /// Gets or sets Plugin Version. + /// + [JsonPropertyName("version")] + public string? Version { get; set; } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/AccessedResourceDetails.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/AccessedResourceDetails.cs new file mode 100644 index 0000000000..e9a18543c6 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/AccessedResourceDetails.cs @@ -0,0 +1,53 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +namespace Microsoft.Agents.AI.Purview.Models.Common; + +/// +/// Information about a resource accessed during a conversation. +/// +internal sealed class AccessedResourceDetails +{ + /// + /// Resource ID. + /// + [JsonPropertyName("identifier")] + public string? Identifier { get; set; } + + /// + /// Resource name. + /// + [JsonPropertyName("name")] + public string? Name { get; set; } + + /// + /// Resource URL. + /// + [JsonPropertyName("url")] + public string? Url { get; set; } + + /// + /// Sensitivity label id detected on the resource. + /// + [JsonPropertyName("labelId")] + public string? LabelId { get; set; } + + /// + /// Access type performed on the resource. + /// + [JsonPropertyName("accessType")] + public ResourceAccessType AccessType { get; set; } + + /// + /// Status of the access operation. + /// + [JsonPropertyName("status")] + public ResourceAccessStatus Status { get; set; } + + /// + /// Indicates if cross prompt injection was detected. + /// + [JsonPropertyName("isCrossPromptInjectionDetected")] + public bool? IsCrossPromptInjectionDetected { get; set; } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/Activity.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/Activity.cs new file mode 100644 index 0000000000..5f9fdeb9d7 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/Activity.cs @@ -0,0 +1,44 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Runtime.Serialization; +using System.Text.Json.Serialization; + +namespace Microsoft.Agents.AI.Purview.Models.Common; + +/// +/// Activity definitions +/// +[DataContract] +[JsonConverter(typeof(JsonStringEnumConverter))] +internal enum Activity : int +{ + /// + /// Unknown activity + /// + [EnumMember(Value = "unknown")] + Unknown = 0, + + /// + /// Upload text + /// + [EnumMember(Value = "uploadText")] + UploadText = 1, + + /// + /// Upload file + /// + [EnumMember(Value = "uploadFile")] + UploadFile = 2, + + /// + /// Download text + /// + [EnumMember(Value = "downloadText")] + DownloadText = 3, + + /// + /// Download file + /// + [EnumMember(Value = "downloadFile")] + DownloadFile = 4, +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ActivityMetadata.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ActivityMetadata.cs new file mode 100644 index 0000000000..deefc24560 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ActivityMetadata.cs @@ -0,0 +1,30 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Runtime.Serialization; +using System.Text.Json.Serialization; + +namespace Microsoft.Agents.AI.Purview.Models.Common; + +/// +/// Request for metadata information +/// +[DataContract] +internal sealed class ActivityMetadata +{ + /// + /// Initializes a new instance of the class. + /// + /// The activity performed with the content. + public ActivityMetadata(Activity activity) + { + this.Activity = activity; + } + + /// + /// The activity performed with the content. + /// + [DataMember] + [JsonConverter(typeof(JsonStringEnumConverter))] + [JsonPropertyName("activity")] + public Activity Activity { get; } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ClassificationErrorBase.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ClassificationErrorBase.cs new file mode 100644 index 0000000000..e52bf9ebb4 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ClassificationErrorBase.cs @@ -0,0 +1,37 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +namespace Microsoft.Agents.AI.Purview.Models.Common; + +/// +/// Base error contract returned when some exception occurs. +/// +[JsonDerivedType(typeof(ProcessingError))] +internal class ClassificationErrorBase +{ + /// + /// Gets or sets the error code. + /// + [JsonPropertyName("code")] + public string? ErrorCode { get; set; } + + /// + /// Gets or sets the message. + /// + [JsonPropertyName("message")] + public string? Message { get; set; } + + /// + /// Gets or sets target of error. + /// + [JsonPropertyName("target")] + public string? Target { get; set; } + + /// + /// Gets or sets an object containing more specific information than the current object about the error. + /// It can't be a Dictionary because OData will make ClassificationErrorBase open type. It's not expected behavior. + /// + [JsonPropertyName("innerError")] + public ClassificationInnerError? InnerError { get; set; } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ClassificationInnerError.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ClassificationInnerError.cs new file mode 100644 index 0000000000..1133529188 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ClassificationInnerError.cs @@ -0,0 +1,36 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Text.Json.Serialization; + +namespace Microsoft.Agents.AI.Purview.Models.Common; + +/// +/// Inner classification error. +/// +internal sealed class ClassificationInnerError +{ + /// + /// Gets or sets date of error. + /// + [JsonPropertyName("date")] + public DateTime? Date { get; set; } + + /// + /// Gets or sets error code. + /// + [JsonPropertyName("code")] + public string? ErrorCode { get; set; } + + /// + /// Gets or sets client request ID. + /// + [JsonPropertyName("clientRequestId")] + public string? ClientRequestId { get; set; } + + /// + /// Gets or sets Activity ID. + /// + [JsonPropertyName("activityId")] + public string? ActivityId { get; set; } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ContentBase.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ContentBase.cs new file mode 100644 index 0000000000..6a2a92226d --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ContentBase.cs @@ -0,0 +1,21 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +namespace Microsoft.Agents.AI.Purview.Models.Common; + +/// +/// Base class for content items to be processed by the Purview SDK. +/// +[JsonDerivedType(typeof(PurviewTextContent))] +[JsonDerivedType(typeof(PurviewBinaryContent))] +internal abstract class ContentBase : GraphDataTypeBase +{ + /// + /// Creates a new instance of the class. + /// + /// The graph data type of the content. + protected ContentBase(string dataType) : base(dataType) + { + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ContentProcessingErrorType.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ContentProcessingErrorType.cs new file mode 100644 index 0000000000..3d57a02aee --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ContentProcessingErrorType.cs @@ -0,0 +1,27 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +namespace Microsoft.Agents.AI.Purview.Models.Common; + +/// +/// Type of error that occurred during content processing. +/// +[JsonConverter(typeof(JsonStringEnumConverter))] +internal enum ContentProcessingErrorType +{ + /// + /// Error is transient. + /// + Transient, + + /// + /// Error is permanent. + /// + Permanent, + + /// + /// Unknown future value placeholder. + /// + UnknownFutureValue +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ContentToProcess.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ContentToProcess.cs new file mode 100644 index 0000000000..9e2e5824f3 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ContentToProcess.cs @@ -0,0 +1,70 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Collections.Generic; +using System.Runtime.Serialization; +using System.Text.Json.Serialization; + +namespace Microsoft.Agents.AI.Purview.Models.Common; + +/// +/// Content to be processed by process content. +/// +internal sealed class ContentToProcess +{ + /// + /// Creates a new instance of ContentToProcess. + /// + /// The content to send and its associated ids. + /// Metadata about the activity performed with the content. + /// Metadata about the device that produced the content. + /// Metadata about the application integrating with Purview. + /// Metadata about the application being protected by Purview. + public ContentToProcess( + List contentEntries, + ActivityMetadata activityMetadata, + DeviceMetadata deviceMetadata, + IntegratedAppMetadata integratedAppMetadata, + ProtectedAppMetadata protectedAppMetadata) + { + this.ContentEntries = contentEntries; + this.ActivityMetadata = activityMetadata; + this.DeviceMetadata = deviceMetadata; + this.IntegratedAppMetadata = integratedAppMetadata; + this.ProtectedAppMetadata = protectedAppMetadata; + } + + /// + /// Gets or sets the content entries. + /// List of activities supported by caller. It is used to trim response to activities interesting to the caller. + /// + [JsonPropertyName("contentEntries")] + public List ContentEntries { get; set; } + + /// + /// Activity metadata + /// + [DataMember] + [JsonPropertyName("activityMetadata")] + public ActivityMetadata ActivityMetadata { get; set; } + + /// + /// Device metadata + /// + [DataMember] + [JsonPropertyName("deviceMetadata")] + public DeviceMetadata DeviceMetadata { get; set; } + + /// + /// Integrated app metadata + /// + [DataMember] + [JsonPropertyName("integratedAppMetadata")] + public IntegratedAppMetadata IntegratedAppMetadata { get; set; } + + /// + /// Protected app metadata + /// + [DataMember] + [JsonPropertyName("protectedAppMetadata")] + public ProtectedAppMetadata ProtectedAppMetadata { get; set; } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/DeviceMetadata.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/DeviceMetadata.cs new file mode 100644 index 0000000000..3a60686be3 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/DeviceMetadata.cs @@ -0,0 +1,29 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +namespace Microsoft.Agents.AI.Purview.Models.Common; + +/// +/// Endpoint device Metdata +/// +internal sealed class DeviceMetadata +{ + /// + /// Device type + /// + [JsonPropertyName("deviceType")] + public string? DeviceType { get; set; } + + /// + /// The ip address of the device. + /// + [JsonPropertyName("ipAddress")] + public string? IpAddress { get; set; } + + /// + /// OS specifications + /// + [JsonPropertyName("operatingSystemSpecifications")] + public OperatingSystemSpecifications? OperatingSystemSpecifications { get; set; } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/DlpAction.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/DlpAction.cs new file mode 100644 index 0000000000..8eda013588 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/DlpAction.cs @@ -0,0 +1,69 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +namespace Microsoft.Agents.AI.Purview.Models.Common; + +/// +/// Defines all the actions for DLP. +/// +[JsonConverter(typeof(JsonStringEnumConverter))] +internal enum DlpAction +{ + /// + /// The DLP action to notify user. + /// + NotifyUser, + + /// + /// The DLP action is block. + /// + BlockAccess, + + /// + /// The DLP action to apply restrictions on device. + /// + DeviceRestriction, + + /// + /// The DLP action to apply restrictions on browsers. + /// + BrowserRestriction, + + /// + /// The DLP action to generate an alert + /// + GenerateAlert, + + /// + /// The DLP action to generate an incident report + /// + GenerateIncidentReportAction, + + /// + /// The DLP action to block anonymous link access in SPO + /// + SPBlockAnonymousAccess, + + /// + /// DLP Action to disallow guest access in SPO + /// + SPRuntimeAccessControl, + + /// + /// DLP No Op action for NotifyUser. Used in Block Access V2 rule + /// + SPSharingNotifyUser, + + /// + /// DLP No Op action for GIR. Used in Block Access V2 rule + /// + SPSharingGenerateIncidentReport, + + /// + /// Restrict access action for data in motion scenarios. + /// Advanced version of BlockAccess which can take both enforced restriction mode (Audit, Block, etc.) + /// and action triggers (Print, SaveToLocal, etc.) as parameters. + /// + RestrictAccess, +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/DlpActionInfo.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/DlpActionInfo.cs new file mode 100644 index 0000000000..a5846acadc --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/DlpActionInfo.cs @@ -0,0 +1,23 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +namespace Microsoft.Agents.AI.Purview.Models.Common; + +/// +/// Base class to define DLP Actions. +/// +internal sealed class DlpActionInfo +{ + /// + /// Gets or sets the type of the DLP action. + /// + [JsonPropertyName("action")] + public DlpAction Action { get; set; } + + /// + /// The type of restriction action to take. + /// + [JsonPropertyName("restrictionAction")] + public RestrictionAction? RestrictionAction { get; set; } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ErrorDetails.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ErrorDetails.cs new file mode 100644 index 0000000000..dd79ee13ce --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ErrorDetails.cs @@ -0,0 +1,23 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +namespace Microsoft.Agents.AI.Purview.Models.Common; + +/// +/// Represents the details of an error. +/// +internal sealed class ErrorDetails +{ + /// + /// Gets or sets the error code. + /// + [JsonPropertyName("code")] + public string? Code { get; set; } + + /// + /// Gets or sets the error message. + /// + [JsonPropertyName("message")] + public string? Message { get; set; } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ExecutionMode.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ExecutionMode.cs new file mode 100644 index 0000000000..3fecfbb3f4 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ExecutionMode.cs @@ -0,0 +1,27 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +namespace Microsoft.Agents.AI.Purview.Models.Common; + +/// +/// Request execution mode +/// +[JsonConverter(typeof(JsonStringEnumConverter))] +internal enum ExecutionMode : int +{ + /// + /// Evaluate inline. + /// + EvaluateInline = 1, + + /// + /// Evaluate offline. + /// + EvaluateOffline = 2, + + /// + /// Unknown future value. + /// + UnknownFutureValue = 3 +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/GraphDataTypeBase.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/GraphDataTypeBase.cs new file mode 100644 index 0000000000..df54240662 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/GraphDataTypeBase.cs @@ -0,0 +1,26 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +namespace Microsoft.Agents.AI.Purview.Models.Common; + +/// +/// Base class for all graph data types used in the Purview SDK. +/// +internal abstract class GraphDataTypeBase +{ + /// + /// Create a new instance of the class. + /// + /// The data type of the graph object. + protected GraphDataTypeBase(string dataType) + { + this.DataType = dataType; + } + + /// + /// The @odata.type property name used in the JSON representation of the object. + /// + [JsonPropertyName(Constants.ODataTypePropertyName)] + public string DataType { get; set; } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/IntegratedAppMetadata.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/IntegratedAppMetadata.cs new file mode 100644 index 0000000000..1a5e8b5e13 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/IntegratedAppMetadata.cs @@ -0,0 +1,27 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Runtime.Serialization; +using System.Text.Json.Serialization; + +namespace Microsoft.Agents.AI.Purview.Models.Common; + +/// +/// Request for metadata information +/// +[JsonDerivedType(typeof(ProtectedAppMetadata))] +internal class IntegratedAppMetadata +{ + /// + /// Application name + /// + [DataMember] + [JsonPropertyName("name")] + public string? Name { get; set; } + + /// + /// Application version + /// + [DataMember] + [JsonPropertyName("version")] + public string? Version { get; set; } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/OperatingSystemSpecifications.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/OperatingSystemSpecifications.cs new file mode 100644 index 0000000000..3ea8837177 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/OperatingSystemSpecifications.cs @@ -0,0 +1,23 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +namespace Microsoft.Agents.AI.Purview.Models.Common; + +/// +/// Operating System Specifications +/// +internal sealed class OperatingSystemSpecifications +{ + /// + /// OS platform + /// + [JsonPropertyName("operatingSystemPlatform")] + public string? OperatingSystemPlatform { get; set; } + + /// + /// OS version + /// + [JsonPropertyName("operatingSystemVersion")] + public string? OperatingSystemVersion { get; set; } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/PolicyBinding.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/PolicyBinding.cs new file mode 100644 index 0000000000..9898f62e01 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/PolicyBinding.cs @@ -0,0 +1,25 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Collections.Generic; +using System.Text.Json.Serialization; + +namespace Microsoft.Agents.AI.Purview.Models.Common; + +/// +/// Represents user scoping information, i.e. which users are affected by the policy. +/// +internal sealed class PolicyBinding +{ + /// + /// Gets or sets the users to be included. + /// + [JsonPropertyName("inclusions")] + public ICollection? Inclusions { get; set; } + + /// + /// Gets or sets the users to be excluded. + /// Exclusions may not be present in the response, thus this property is nullable. + /// + [JsonPropertyName("exclusions")] + public ICollection? Exclusions { get; set; } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/PolicyLocation.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/PolicyLocation.cs new file mode 100644 index 0000000000..c0a40974e5 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/PolicyLocation.cs @@ -0,0 +1,27 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +namespace Microsoft.Agents.AI.Purview.Models.Common; + +/// +/// Represents a location to which policy is applicable. +/// +internal sealed class PolicyLocation : GraphDataTypeBase +{ + /// + /// Creates a new instance of the class. + /// + /// The graph data type of the PolicyLocation object. + /// THe value of the policy location: app id, domain, etc. + public PolicyLocation(string dataType, string value) : base(dataType) + { + this.Value = value; + } + + /// + /// Gets or sets the applicable value for location. + /// + [JsonPropertyName("value")] + public string Value { get; set; } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/PolicyPivotProperty.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/PolicyPivotProperty.cs new file mode 100644 index 0000000000..d56a374842 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/PolicyPivotProperty.cs @@ -0,0 +1,42 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Runtime.Serialization; +using System.Text.Json.Serialization; + +namespace Microsoft.Agents.AI.Purview.Models.Common; + +/// +/// Property for policy scoping response to aggregate on +/// +[DataContract] +[JsonConverter(typeof(JsonStringEnumConverter))] +internal enum PolicyPivotProperty : int +{ + /// + /// Unknown activity + /// + [EnumMember] + [JsonPropertyName("none")] + None = 0, + + /// + /// Pivot on Activity + /// + [EnumMember] + [JsonPropertyName("activity")] + Activity = 1, + + /// + /// Pivot on location + /// + [EnumMember] + [JsonPropertyName("location")] + Location = 2, + + /// + /// Pivot on location + /// + [EnumMember] + [JsonPropertyName("unknownFutureValue")] + UnknownFutureValue = 3, +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/PolicyScope.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/PolicyScope.cs new file mode 100644 index 0000000000..f00e941d35 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/PolicyScope.cs @@ -0,0 +1,43 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Collections.Generic; +using System.Text.Json.Serialization; + +namespace Microsoft.Agents.AI.Purview.Models.Common; + +/// +/// Represents a scope for policy protection. +/// +internal sealed class PolicyScopeBase +{ + /// + /// Gets or sets the locations to be protected, e.g. domains or URLs. + /// + [JsonPropertyName("locations")] + public ICollection? Locations { get; set; } + + /// + /// Gets or sets the activities to be protected, e.g. uploadText, downloadText. + /// + [JsonPropertyName("activities")] + public ProtectionScopeActivities Activities { get; set; } + + /// + /// Gets or sets how policy should be executed - fire-and-forget or wait for completion. + /// + [JsonPropertyName("executionMode")] + public ExecutionMode ExecutionMode { get; set; } + + /// + /// Gets or sets the enforcement actions to be taken on activities and locations from this scope. + /// There may be no actions in the response. + /// + [JsonPropertyName("policyActions")] + public ICollection? PolicyActions { get; set; } + + /// + /// Gets or sets information about policy applicability to a specific user. + /// + [JsonPropertyName("policyScope")] + public PolicyBinding? PolicyScope { get; set; } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ProcessContentMetadataBase.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ProcessContentMetadataBase.cs new file mode 100644 index 0000000000..a401288127 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ProcessContentMetadataBase.cs @@ -0,0 +1,94 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Text.Json.Serialization; + +namespace Microsoft.Agents.AI.Purview.Models.Common; + +/// +/// Base class for process content metadata. +/// +[JsonDerivedType(typeof(ProcessConversationMetadata))] +[JsonDerivedType(typeof(ProcessFileMetadata))] +internal abstract class ProcessContentMetadataBase : GraphDataTypeBase +{ + private const string ProcessConversationMetadataDataType = Constants.ODataGraphNamespace + ".processConversationMetadata"; + + /// + /// Creates a new instance of ProcessContentMetadataBase. + /// + /// The content that will be processed. + /// The unique identifier for the content. + /// Indicates if the content is truncated. + /// The name of the content. + protected ProcessContentMetadataBase(ContentBase content, string identifier, bool isTruncated, string name) : base(ProcessConversationMetadataDataType) + { + this.Identifier = identifier; + this.IsTruncated = isTruncated; + this.Content = content; + this.Name = name; + } + + /// + /// Gets or sets the identifier. + /// Unique id for the content. It is specific to the enforcement plane. Path is used as item unique identifier, e.g., guid of a message in the conversation, file URL, storage file path, message ID, etc. + /// + [JsonPropertyName("identifier")] + public string Identifier { get; set; } + + /// + /// Gets or sets the content. + /// The content to be processed. + /// + [JsonPropertyName("content")] + public ContentBase Content { get; set; } + + /// + /// Gets or sets the name. + /// Name of the content, e.g., file name or web page title. + /// + [JsonPropertyName("name")] + public string Name { get; set; } + + /// + /// Gets or sets the correlationId. + /// Identifier to group multiple contents. + /// + [JsonPropertyName("correlationId")] + public string? CorrelationId { get; set; } + + /// + /// Gets or sets the sequenceNumber. + /// Sequence in which the content was originally generated. + /// + [JsonPropertyName("sequenceNumber")] + public long? SequenceNumber { get; set; } + + /// + /// Gets or sets the length. + /// Content length in bytes. + /// + [JsonPropertyName("length")] + public long? Length { get; set; } + + /// + /// Gets or sets the isTruncated. + /// Indicates if the original content has been truncated, e.g., to meet text or file size limits. + /// + [JsonPropertyName("isTruncated")] + public bool IsTruncated { get; set; } + + /// + /// Gets or sets the createdDateTime. + /// When the content was created. E.g., file created time or the time when a message was sent. + /// + [JsonPropertyName("createdDateTime")] + public DateTimeOffset CreatedDateTime { get; set; } = DateTime.UtcNow; + + /// + /// Gets or sets the modifiedDateTime. + /// When the content was last modified. E.g., file last modified time. For content created on the fly, such as messaging, whenModified and whenCreated are expected to be the same. + /// + [JsonPropertyName("modifiedDateTime")] + public DateTimeOffset? ModifiedDateTime { get; set; } = DateTime.UtcNow; +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ProcessConversationMetadata.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ProcessConversationMetadata.cs new file mode 100644 index 0000000000..86bedb9248 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ProcessConversationMetadata.cs @@ -0,0 +1,46 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Collections.Generic; +using System.Text.Json.Serialization; + +namespace Microsoft.Agents.AI.Purview.Models.Common; + +/// +/// Represents metadata for conversation content to be processed by the Purview SDK. +/// +internal sealed class ProcessConversationMetadata : ProcessContentMetadataBase +{ + private const string ProcessConversationMetadataDataType = Constants.ODataGraphNamespace + ".processConversationMetadata"; + + /// + /// Initializes a new instance of the class. + /// + public ProcessConversationMetadata(ContentBase contentBase, string identifier, bool isTruncated, string name) : base(contentBase, identifier, isTruncated, name) + { + this.DataType = ProcessConversationMetadataDataType; + } + + /// + /// Gets or sets the parent message ID for nested conversations. + /// + [JsonPropertyName("parentMessageId")] + public string? ParentMessageId { get; set; } + + /// + /// Gets or sets the accessed resources during message generation for bot messages. + /// + [JsonPropertyName("accessedResources_v2")] + public List? AccessedResources { get; set; } + + /// + /// Gets or sets the plugins used during message generation for bot messages. + /// + [JsonPropertyName("plugins")] + public List? Plugins { get; set; } + + /// + /// Gets or sets the collection of AI agent information. + /// + [JsonPropertyName("agents")] + public List? Agents { get; set; } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ProcessFileMetadata.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ProcessFileMetadata.cs new file mode 100644 index 0000000000..a9f1749bed --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ProcessFileMetadata.cs @@ -0,0 +1,27 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +namespace Microsoft.Agents.AI.Purview.Models.Common; + +/// +/// Represents metadata for a file content to be processed by the Purview SDK. +/// +internal sealed class ProcessFileMetadata : ProcessContentMetadataBase +{ + private const string ProcessFileMetadataDataType = Constants.ODataGraphNamespace + ".processFileMetadata"; + + /// + /// Initializes a new instance of the class. + /// + public ProcessFileMetadata(ContentBase contentBase, string identifier, bool isTruncated, string name) : base(contentBase, identifier, isTruncated, name) + { + this.DataType = ProcessFileMetadataDataType; + } + + /// + /// Gets or sets the owner ID. + /// + [JsonPropertyName("ownerId")] + public string? OwnerId { get; set; } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ProcessingError.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ProcessingError.cs new file mode 100644 index 0000000000..4852d5ca8a --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ProcessingError.cs @@ -0,0 +1,24 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Collections.Generic; +using System.Text.Json.Serialization; + +namespace Microsoft.Agents.AI.Purview.Models.Common; + +/// +/// Contains information about a processing error. +/// +internal sealed class ProcessingError : ClassificationErrorBase +{ + /// + /// Details about the error. + /// + [JsonPropertyName("details")] + public List? Details { get; set; } + + /// + /// Gets or sets the error type. + /// + [JsonPropertyName("type")] + public ContentProcessingErrorType? Type { get; set; } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ProtectedAppMetadata.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ProtectedAppMetadata.cs new file mode 100644 index 0000000000..984a4168e7 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ProtectedAppMetadata.cs @@ -0,0 +1,26 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +namespace Microsoft.Agents.AI.Purview.Models.Common; + +/// +/// Represents metadata for a protected application that is integrated with Purview. +/// +internal sealed class ProtectedAppMetadata : IntegratedAppMetadata +{ + /// + /// Creates a new instance of the class. + /// + /// The location information of the protected app's data. + public ProtectedAppMetadata(PolicyLocation applicationLocation) + { + this.ApplicationLocation = applicationLocation; + } + + /// + /// The location of the application. + /// + [JsonPropertyName("applicationLocation")] + public PolicyLocation ApplicationLocation { get; set; } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ProtectionScopeActivities.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ProtectionScopeActivities.cs new file mode 100644 index 0000000000..6c93a76124 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ProtectionScopeActivities.cs @@ -0,0 +1,52 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Runtime.Serialization; +using System.Text.Json.Serialization; + +namespace Microsoft.Agents.AI.Purview.Models.Common; + +/// +/// Activities that can be protected by the Purview Protection Scopes API. +/// +[Flags] +[DataContract] +[JsonConverter(typeof(JsonStringEnumConverter))] +internal enum ProtectionScopeActivities +{ + /// + /// None. + /// + [EnumMember(Value = "none")] + None = 0, + + /// + /// Upload text activity. + /// + [EnumMember(Value = "uploadText")] + UploadText = 1, + + /// + /// Upload file activity. + /// + [EnumMember(Value = "uploadFile")] + UploadFile = 2, + + /// + /// Download text activity. + /// + [EnumMember(Value = "downloadText")] + DownloadText = 4, + + /// + /// Download file activity. + /// + [EnumMember(Value = "downloadFile")] + DownloadFile = 8, + + /// + /// Unknown future value. + /// + [EnumMember(Value = "unknownFutureValue")] + UnknownFutureValue = 16 +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ProtectionScopeState.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ProtectionScopeState.cs new file mode 100644 index 0000000000..8fc7a534ad --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ProtectionScopeState.cs @@ -0,0 +1,27 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +namespace Microsoft.Agents.AI.Purview.Models.Common; + +/// +/// Indicates status of protection scope changes. +/// +[JsonConverter(typeof(JsonStringEnumConverter))] +internal enum ProtectionScopeState +{ + /// + /// Scope state hasn't changed. + /// + NotModified = 0, + + /// + /// Scope state has changed. + /// + Modified = 1, + + /// + /// Unknown value placeholder for future use. + /// + UnknownFutureValue = 2 +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ProtectionScopesCacheKey.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ProtectionScopesCacheKey.cs new file mode 100644 index 0000000000..2c772cbcb0 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ProtectionScopesCacheKey.cs @@ -0,0 +1,91 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Linq; +using Microsoft.Agents.AI.Purview.Models.Requests; + +namespace Microsoft.Agents.AI.Purview.Models.Common; + +/// +/// A cache key for storing protection scope responses. +/// +internal sealed class ProtectionScopesCacheKey +{ + /// + /// Creates a new instance of . + /// + /// The entra id of the user who made the interaction. + /// The tenant id of the user who made the interaction. + /// The activity performed with the data. + /// The location where the data came from. + /// The property to pivot on. + /// Metadata about the device that made the interaction. + /// Metadata about the app that is integrating with Purview. + public ProtectionScopesCacheKey( + string userId, + string tenantId, + ProtectionScopeActivities activities, + PolicyLocation? location, + PolicyPivotProperty? pivotOn, + DeviceMetadata? deviceMetadata, + IntegratedAppMetadata? integratedAppMetadata) + { + this.UserId = userId; + this.TenantId = tenantId; + this.Activities = activities; + this.Location = location; + this.PivotOn = pivotOn; + this.DeviceMetadata = deviceMetadata; + this.IntegratedAppMetadata = integratedAppMetadata; + } + + /// + /// Creates a mew instance of . + /// + /// A protection scopes request. + public ProtectionScopesCacheKey( + ProtectionScopesRequest request) : this( + request.UserId, + request.TenantId, + request.Activities, + request.Locations.FirstOrDefault(), + request.PivotOn, + request.DeviceMetadata, + request.IntegratedAppMetadata) + { + } + + /// + /// The id of the user making the request. + /// + public string UserId { get; set; } + + /// + /// The id of the tenant containing the user making the request. + /// + public string TenantId { get; set; } + + /// + /// The activity performed with the content. + /// + public ProtectionScopeActivities Activities { get; set; } + + /// + /// The location of the application. + /// + public PolicyLocation? Location { get; set; } + + /// + /// The property used to pivot the policy evaluation. + /// + public PolicyPivotProperty? PivotOn { get; set; } + + /// + /// Metadata about the device used to access the content. + /// + public DeviceMetadata? DeviceMetadata { get; set; } + + /// + /// Metadata about the integrated app used to access the content. + /// + public IntegratedAppMetadata? IntegratedAppMetadata { get; set; } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/PurviewBinaryContent.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/PurviewBinaryContent.cs new file mode 100644 index 0000000000..0d65ac341d --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/PurviewBinaryContent.cs @@ -0,0 +1,28 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +namespace Microsoft.Agents.AI.Purview.Models.Common; + +/// +/// Represents a binary content item to be processed. +/// +internal sealed class PurviewBinaryContent : ContentBase +{ + private const string BinaryContentDataType = Constants.ODataGraphNamespace + ".binaryContent"; + + /// + /// Initializes a new instance of the class. + /// + /// The binary content in byte array format. + public PurviewBinaryContent(byte[] data) : base(BinaryContentDataType) + { + this.Data = data; + } + + /// + /// Gets or sets the binary data. + /// + [JsonPropertyName("data")] + public byte[] Data { get; set; } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/PurviewTextContent.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/PurviewTextContent.cs new file mode 100644 index 0000000000..cfd03ae6ce --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/PurviewTextContent.cs @@ -0,0 +1,28 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +namespace Microsoft.Agents.AI.Purview.Models.Common; + +/// +/// Represents a text content item to be processed. +/// +internal sealed class PurviewTextContent : ContentBase +{ + private const string TextContentDataType = Constants.ODataGraphNamespace + ".textContent"; + + /// + /// Initializes a new instance of the class. + /// + /// The text content in string format. + public PurviewTextContent(string data) : base(TextContentDataType) + { + this.Data = data; + } + + /// + /// Gets or sets the text data. + /// + [JsonPropertyName("data")] + public string Data { get; set; } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ResourceAccessStatus.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ResourceAccessStatus.cs new file mode 100644 index 0000000000..623f138e8b --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ResourceAccessStatus.cs @@ -0,0 +1,32 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Runtime.Serialization; +using System.Text.Json.Serialization; + +namespace Microsoft.Agents.AI.Purview.Models.Common; + +/// +/// Status of the access operation. +/// +[DataContract] +[JsonConverter(typeof(JsonStringEnumConverter))] +internal enum ResourceAccessStatus +{ + /// + /// Represents failed access to the resource. + /// + [EnumMember(Value = "failure")] + Failure = 0, + + /// + /// Represents successful access to the resource. + /// + [EnumMember(Value = "success")] + Success = 1, + + /// + /// Unknown future value. + /// + [EnumMember(Value = "unknownFutureValue")] + UnknownFutureValue = 2 +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ResourceAccessType.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ResourceAccessType.cs new file mode 100644 index 0000000000..cb4e3b0cab --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/ResourceAccessType.cs @@ -0,0 +1,46 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Runtime.Serialization; +using System.Text.Json.Serialization; + +namespace Microsoft.Agents.AI.Purview.Models.Common; + +/// +/// Access type performed on the resource. +/// +[Flags] +[DataContract] +[JsonConverter(typeof(JsonStringEnumConverter))] +internal enum ResourceAccessType : long +{ + /// + /// No access type. + /// + [EnumMember(Value = "none")] + None = 0, + + /// + /// Read access. + /// + [EnumMember(Value = "read")] + Read = 1 << 0, + + /// + /// Write access. + /// + [EnumMember(Value = "write")] + Write = 1 << 1, + + /// + /// Create access. + /// + [EnumMember(Value = "create")] + Create = 1 << 2, + + /// + /// Unknown future value. + /// + [EnumMember(Value = "unknownFutureValue")] + UnknownFutureValue = 1 << 3 +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/RestrictionAction.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/RestrictionAction.cs new file mode 100644 index 0000000000..ea13ec36a6 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/RestrictionAction.cs @@ -0,0 +1,32 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +namespace Microsoft.Agents.AI.Purview.Models.Common; + +/// +/// Restriction actions for devices. +/// +[JsonConverter(typeof(JsonStringEnumConverter))] +internal enum RestrictionAction +{ + /// + /// Warn Action. + /// + Warn, + + /// + /// Audit action. + /// + Audit, + + /// + /// Block action. + /// + Block, + + /// + /// Allow action + /// + Allow +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/Scope.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/Scope.cs new file mode 100644 index 0000000000..9fc4de38fe --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/Scope.cs @@ -0,0 +1,23 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; + +namespace Microsoft.Agents.AI.Purview.Models.Common; + +/// +/// Represents tenant/user/group scopes. +/// +internal sealed class Scope +{ + /// + /// The odata type of the scope used to identify what type of scope was returned. + /// + [JsonPropertyName("@odata.type")] + public string? ODataType { get; set; } + + /// + /// Gets or sets the scope identifier. + /// + [JsonPropertyName("identity")] + public string? Identity { get; set; } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/TokenInfo.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/TokenInfo.cs new file mode 100644 index 0000000000..bd1338dd64 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Common/TokenInfo.cs @@ -0,0 +1,29 @@ +// Copyright (c) Microsoft. All rights reserved. + +namespace Microsoft.Agents.AI.Purview.Models.Common; + +/// +/// Info pulled from an auth token. +/// +internal sealed class TokenInfo +{ + /// + /// The entra id of the authenticated user. This is null if the auth token is not a user token. + /// + public string? UserId { get; set; } + + /// + /// The tenant id of the auth token. + /// + public string? TenantId { get; set; } + + /// + /// The client id of the auth token. + /// + public string? ClientId { get; set; } + + /// + /// Gets a value indicating whether the token is associated with a user. + /// + public bool IsUserToken => !string.IsNullOrEmpty(this.UserId); +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Models/Jobs/BackgroundJobBase.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Jobs/BackgroundJobBase.cs new file mode 100644 index 0000000000..d3c9317628 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Jobs/BackgroundJobBase.cs @@ -0,0 +1,8 @@ +// Copyright (c) Microsoft. All rights reserved. + +namespace Microsoft.Agents.AI.Purview.Models.Jobs; + +/// +/// Abstract base class for background jobs. +/// +internal abstract class BackgroundJobBase; diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Models/Jobs/ContentActivityJob.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Jobs/ContentActivityJob.cs new file mode 100644 index 0000000000..513af7f331 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Jobs/ContentActivityJob.cs @@ -0,0 +1,25 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Microsoft.Agents.AI.Purview.Models.Requests; + +namespace Microsoft.Agents.AI.Purview.Models.Jobs; + +/// +/// Class representing a job to send content activities to the Purview service. +/// +internal sealed class ContentActivityJob : BackgroundJobBase +{ + /// + /// Create a new instance of the class. + /// + /// The content activities request to be sent in the background. + public ContentActivityJob(ContentActivitiesRequest request) + { + this.Request = request; + } + + /// + /// The request to send to the Purview service. + /// + public ContentActivitiesRequest Request { get; } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Models/Jobs/ProcessContentJob.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Jobs/ProcessContentJob.cs new file mode 100644 index 0000000000..768588f9d7 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Jobs/ProcessContentJob.cs @@ -0,0 +1,25 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Microsoft.Agents.AI.Purview.Models.Requests; + +namespace Microsoft.Agents.AI.Purview.Models.Jobs; + +/// +/// Class representing a job to process content. +/// +internal sealed class ProcessContentJob : BackgroundJobBase +{ + /// + /// Initializes a new instance of the class. + /// + /// The process content request to be sent in the background. + public ProcessContentJob(ProcessContentRequest request) + { + this.Request = request; + } + + /// + /// The request to process content. + /// + public ProcessContentRequest Request { get; } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Models/Requests/ContentActivitiesRequest.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Requests/ContentActivitiesRequest.cs new file mode 100644 index 0000000000..a754a5a56f --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Requests/ContentActivitiesRequest.cs @@ -0,0 +1,66 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Text.Json.Serialization; +using Microsoft.Agents.AI.Purview.Models.Common; + +namespace Microsoft.Agents.AI.Purview.Models.Requests; + +/// +/// A request class used for contentActivity requests. +/// +internal sealed class ContentActivitiesRequest +{ + /// + /// Initializes a new instance of the class. + /// + /// The entra id of the user who performed the activity. + /// The tenant id of the user who performed the activity. + /// The metadata about the content that was sent. + /// The correlation id of the request. + /// The scope identifier of the protection scopes associated with this request. + public ContentActivitiesRequest(string userId, string tenantId, ContentToProcess contentMetadata, Guid correlationId = default, string? scopeIdentifier = null) + { + this.UserId = userId ?? throw new ArgumentNullException(nameof(userId)); + this.TenantId = tenantId ?? throw new ArgumentNullException(nameof(tenantId)); + this.ContentMetadata = contentMetadata ?? throw new ArgumentNullException(nameof(contentMetadata)); + this.CorrelationId = correlationId == default ? Guid.NewGuid() : correlationId; + this.ScopeIdentifier = scopeIdentifier; + } + + /// + /// Gets or sets the ID of the signal. + /// + [JsonPropertyName("id")] + public string Id { get; set; } = Guid.NewGuid().ToString(); + + /// + /// Gets or sets the user ID of the content that is generating the signal. + /// + [JsonPropertyName("userId")] + public string UserId { get; set; } + + /// + /// Gets or sets the scope identifier for the signal. + /// + [JsonPropertyName("scopeIdentifier")] + public string? ScopeIdentifier { get; set; } + + /// + /// Gets or sets the content and associated content metadata for the content used to generate the signal. + /// + [JsonPropertyName("contentMetadata")] + public ContentToProcess ContentMetadata { get; set; } + + /// + /// Gets or sets the correlation ID for the signal. + /// + [JsonIgnore] + public Guid CorrelationId { get; set; } + + /// + /// Gets or sets the tenant id for the signal. + /// + [JsonIgnore] + public string TenantId { get; set; } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Models/Requests/ProcessContentRequest.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Requests/ProcessContentRequest.cs new file mode 100644 index 0000000000..f8e9602cef --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Requests/ProcessContentRequest.cs @@ -0,0 +1,56 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Text.Json.Serialization; +using Microsoft.Agents.AI.Purview.Models.Common; + +namespace Microsoft.Agents.AI.Purview.Models.Requests; + +/// +/// Request for ProcessContent API +/// +internal sealed class ProcessContentRequest +{ + /// + /// Creates a new instance of ProcessContentRequest. + /// + /// The content and its metadata that will be processed. + /// The entra user id of the user making the request. + /// The tenant id of the user making the request. + public ProcessContentRequest(ContentToProcess contentToProcess, string userId, string tenantId) + { + this.ContentToProcess = contentToProcess; + this.UserId = userId; + this.TenantId = tenantId; + } + + /// + /// The content to process. + /// + [JsonPropertyName("contentToProcess")] + public ContentToProcess ContentToProcess { get; set; } + + /// + /// The user id of the user making the request. + /// + [JsonIgnore] + public string UserId { get; set; } + + /// + /// The correlation id of the request. + /// + [JsonIgnore] + public Guid CorrelationId { get; set; } = Guid.NewGuid(); + + /// + /// The tenant id of the user making the request. + /// + [JsonIgnore] + public string TenantId { get; set; } + + /// + /// The identifier of the cached protection scopes. + /// + [JsonIgnore] + internal string? ScopeIdentifier { get; set; } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Models/Requests/ProtectionScopesRequest.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Requests/ProtectionScopesRequest.cs new file mode 100644 index 0000000000..04aba59aff --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Requests/ProtectionScopesRequest.cs @@ -0,0 +1,86 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Collections.Generic; +using System.Runtime.Serialization; +using System.Text.Json.Serialization; +using Microsoft.Agents.AI.Purview.Models.Common; + +namespace Microsoft.Agents.AI.Purview.Models.Requests; + +/// +/// Request model for user protection scopes requests. +/// +[DataContract] +internal sealed class ProtectionScopesRequest +{ + /// + /// Creates a new instance of ProtectionScopesRequest. + /// + /// The entra id of the user who made the interaction. + /// The tenant id of the user who made the interaction. + public ProtectionScopesRequest(string userId, string tenantId) + { + this.UserId = userId; + this.TenantId = tenantId; + } + + /// + /// Activities to include in the scope + /// + [DataMember] + [JsonPropertyName("activities")] + public ProtectionScopeActivities Activities { get; set; } + + /// + /// Gets or sets the locations to compute protection scopes for. + /// + [JsonPropertyName("locations")] + public ICollection Locations { get; set; } = Array.Empty(); + + /// + /// Response aggregation pivot + /// + [DataMember] + [JsonPropertyName("pivotOn")] + public PolicyPivotProperty? PivotOn { get; set; } + + /// + /// Device metadata + /// + [DataMember] + [JsonPropertyName("deviceMetadata")] + public DeviceMetadata? DeviceMetadata { get; set; } + + /// + /// Integrated app metadata + /// + [DataMember] + [JsonPropertyName("integratedAppMetadata")] + public IntegratedAppMetadata? IntegratedAppMetadata { get; set; } + + /// + /// The correlation id of the request. + /// + [JsonIgnore] + public Guid CorrelationId { get; set; } = Guid.NewGuid(); + + /// + /// Scope ID, used to detect stale client scoping information + /// + [DataMember] + [JsonIgnore] + public string ScopeIdentifier { get; set; } = string.Empty; + + /// + /// The id of the user making the request. + /// + [JsonIgnore] + public string UserId { get; set; } + + /// + /// The tenant id of the user making the request. + /// + [JsonIgnore] + public string TenantId { get; set; } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Models/Responses/ContentActivitiesResponse.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Responses/ContentActivitiesResponse.cs new file mode 100644 index 0000000000..afdc21618e --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Responses/ContentActivitiesResponse.cs @@ -0,0 +1,25 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Net; +using System.Text.Json.Serialization; +using Microsoft.Agents.AI.Purview.Models.Common; + +namespace Microsoft.Agents.AI.Purview.Models.Responses; + +/// +/// Represents the response for content activities requests. +/// +internal sealed class ContentActivitiesResponse +{ + /// + /// Gets or sets the HTTP status code associated with the response. + /// + [JsonIgnore] + public HttpStatusCode StatusCode { get; set; } + + /// + /// Details about any errors returned by the request. + /// + [JsonPropertyName("error")] + public ErrorDetails? Error { get; set; } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Models/Responses/ProcessContentResponse.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Responses/ProcessContentResponse.cs new file mode 100644 index 0000000000..c685c7786f --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Responses/ProcessContentResponse.cs @@ -0,0 +1,42 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Collections.Generic; +using System.ComponentModel.DataAnnotations; +using System.Runtime.Serialization; +using System.Text.Json.Serialization; +using Microsoft.Agents.AI.Purview.Models.Common; + +namespace Microsoft.Agents.AI.Purview.Models.Responses; + +/// +/// The response of a process content evaluation. +/// +internal sealed class ProcessContentResponse +{ + /// + /// Gets or sets the evaluation id. + /// + [Key] + public string? Id { get; set; } + + /// + /// Gets or sets the status of protection scope changes. + /// + [DataMember] + [JsonPropertyName("protectionScopeState")] + public ProtectionScopeState? ProtectionScopeState { get; set; } + + /// + /// Gets or sets the policy actions to take. + /// + [DataMember] + [JsonPropertyName("policyActions")] + public IReadOnlyList? PolicyActions { get; set; } + + /// + /// Gets or sets error information about the evaluation. + /// + [DataMember] + [JsonPropertyName("processingErrors")] + public IReadOnlyList? ProcessingErrors { get; set; } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Models/Responses/ProtectionScopesResponse.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Responses/ProtectionScopesResponse.cs new file mode 100644 index 0000000000..fb9b0603d8 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Models/Responses/ProtectionScopesResponse.cs @@ -0,0 +1,24 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Collections.Generic; +using System.Text.Json.Serialization; +using Microsoft.Agents.AI.Purview.Models.Common; + +namespace Microsoft.Agents.AI.Purview.Models.Responses; + +/// +/// A response object containing protection scopes for a tenant. +/// +internal sealed class ProtectionScopesResponse +{ + /// + /// The identifier used for caching the user protection scopes. + /// + public string? ScopeIdentifier { get; set; } + + /// + /// The user protection scopes. + /// + [JsonPropertyName("value")] + public IReadOnlyCollection? Scopes { get; set; } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/PurviewAgent.cs b/dotnet/src/Microsoft.Agents.AI.Purview/PurviewAgent.cs new file mode 100644 index 0000000000..fd2a1950e9 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/PurviewAgent.cs @@ -0,0 +1,70 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Collections.Generic; +using System.Runtime.CompilerServices; +using System.Text.Json; +using System.Threading; +using System.Threading.Tasks; +using Microsoft.Extensions.AI; + +namespace Microsoft.Agents.AI.Purview; + +/// +/// A middleware agent that connects to Microsoft Purview. +/// +internal class PurviewAgent : AIAgent, IDisposable +{ + private readonly AIAgent _innerAgent; + private readonly PurviewWrapper _purviewWrapper; + + /// + /// Initializes a new instance of the class. + /// + /// The agent-framework agent that the middleware wraps. + /// The purview wrapper used to interact with the Purview service. + public PurviewAgent(AIAgent innerAgent, PurviewWrapper purviewWrapper) + { + this._innerAgent = innerAgent; + this._purviewWrapper = purviewWrapper; + } + + /// + public override AgentThread DeserializeThread(JsonElement serializedThread, JsonSerializerOptions? jsonSerializerOptions = null) + { + return this._innerAgent.DeserializeThread(serializedThread, jsonSerializerOptions); + } + + /// + public override AgentThread GetNewThread() + { + return this._innerAgent.GetNewThread(); + } + + /// + public override Task RunAsync(IEnumerable messages, AgentThread? thread = null, AgentRunOptions? options = null, CancellationToken cancellationToken = default) + { + return this._purviewWrapper.ProcessAgentContentAsync(messages, thread, options, this._innerAgent, cancellationToken); + } + + /// + public override async IAsyncEnumerable RunStreamingAsync(IEnumerable messages, AgentThread? thread = null, AgentRunOptions? options = null, [EnumeratorCancellation] CancellationToken cancellationToken = default) + { + var response = await this._purviewWrapper.ProcessAgentContentAsync(messages, thread, options, this._innerAgent, cancellationToken).ConfigureAwait(false); + foreach (var update in response.ToAgentRunResponseUpdates()) + { + yield return update; + } + } + + /// + public void Dispose() + { + if (this._innerAgent is IDisposable disposableAgent) + { + disposableAgent.Dispose(); + } + + this._purviewWrapper.Dispose(); + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/PurviewAppLocation.cs b/dotnet/src/Microsoft.Agents.AI.Purview/PurviewAppLocation.cs new file mode 100644 index 0000000000..0c10db6237 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/PurviewAppLocation.cs @@ -0,0 +1,49 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using Microsoft.Agents.AI.Purview.Models.Common; + +namespace Microsoft.Agents.AI.Purview; + +/// +/// An identifier representing the app's location for Purview policy evaluation. +/// +public class PurviewAppLocation +{ + /// + /// Creates a new instance of . + /// + /// The type of location. + /// The value of the location. + public PurviewAppLocation(PurviewLocationType locationType, string locationValue) + { + this.LocationType = locationType; + this.LocationValue = locationValue; + } + + /// + /// The type of location. + /// + public PurviewLocationType LocationType { get; set; } + + /// + /// The location value. + /// + public string LocationValue { get; set; } + + /// + /// Returns the model for this . + /// + /// PolicyLocation request model. + /// Thrown when an invalid location type is provided. + internal PolicyLocation GetPolicyLocation() + { + return this.LocationType switch + { + PurviewLocationType.Application => new($"{Constants.ODataGraphNamespace}.policyLocationApplication", this.LocationValue), + PurviewLocationType.Uri => new($"{Constants.ODataGraphNamespace}.policyLocationUrl", this.LocationValue), + PurviewLocationType.Domain => new($"{Constants.ODataGraphNamespace}.policyLocationDomain", this.LocationValue), + _ => throw new InvalidOperationException("Invalid location type."), + }; + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/PurviewChatClient.cs b/dotnet/src/Microsoft.Agents.AI.Purview/PurviewChatClient.cs new file mode 100644 index 0000000000..fded26c0ae --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/PurviewChatClient.cs @@ -0,0 +1,62 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Collections.Generic; +using System.Runtime.CompilerServices; +using System.Threading; +using System.Threading.Tasks; +using Microsoft.Extensions.AI; + +namespace Microsoft.Agents.AI.Purview; + +/// +/// A middleware chat client that connects to Microsoft Purview. +/// +internal class PurviewChatClient : IChatClient +{ + private readonly IChatClient _innerChatClient; + private readonly PurviewWrapper _purviewWrapper; + + /// + /// Initializes a new instance of the class. + /// + /// The inner chat client to wrap. + /// The purview wrapper used to interact with the Purview service. + public PurviewChatClient(IChatClient innerChatClient, PurviewWrapper purviewWrapper) + { + this._innerChatClient = innerChatClient; + this._purviewWrapper = purviewWrapper; + } + + /// + public void Dispose() + { + this._purviewWrapper.Dispose(); + this._innerChatClient.Dispose(); + } + + /// + public Task GetResponseAsync(IEnumerable messages, ChatOptions? options = null, CancellationToken cancellationToken = default) + { + return this._purviewWrapper.ProcessChatContentAsync(messages, options, this._innerChatClient, cancellationToken); + } + + /// + public object? GetService(Type serviceType, object? serviceKey = null) + { + return this._innerChatClient.GetService(serviceType, serviceKey); + } + + /// + public async IAsyncEnumerable GetStreamingResponseAsync(IEnumerable messages, + ChatOptions? options = null, + [EnumeratorCancellation] CancellationToken cancellationToken = default) + { + Task responseTask = this._purviewWrapper.ProcessChatContentAsync(messages, options, this._innerChatClient, cancellationToken); + + foreach (var update in (await responseTask.ConfigureAwait(false)).ToChatResponseUpdates()) + { + yield return update; + } + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/PurviewClient.cs b/dotnet/src/Microsoft.Agents.AI.Purview/PurviewClient.cs new file mode 100644 index 0000000000..28013f524e --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/PurviewClient.cs @@ -0,0 +1,323 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Net; +using System.Net.Http; +using System.Text; +using System.Text.Json; +using System.Text.Json.Serialization.Metadata; +using System.Threading; +using System.Threading.Tasks; +using Azure.Core; +using Microsoft.Agents.AI.Purview.Models.Common; +using Microsoft.Agents.AI.Purview.Models.Requests; +using Microsoft.Agents.AI.Purview.Models.Responses; +using Microsoft.Agents.AI.Purview.Serialization; +using Microsoft.Extensions.Logging; +using Microsoft.Extensions.Logging.Abstractions; + +namespace Microsoft.Agents.AI.Purview; + +/// +/// Client for calling Purview APIs. +/// +internal sealed class PurviewClient : IPurviewClient +{ + private readonly TokenCredential _tokenCredential; + private readonly HttpClient _httpClient; + private readonly string[] _scopes; + private readonly string _graphUri; + private readonly ILogger _logger; + + private static PurviewException CreateExceptionForStatusCode(HttpStatusCode statusCode, string endpointName) + { + // .net framework does not support TooManyRequests, so we have to convert to an int. + switch ((int)statusCode) + { + case 429: + return new PurviewRateLimitException($"Rate limit exceeded for {endpointName}."); + case 401: + case 403: + return new PurviewAuthenticationException($"Unauthorized access to {endpointName}. Status code: {statusCode}"); + case 402: + return new PurviewPaymentRequiredException($"Payment required for {endpointName}. Status code: {statusCode}"); + default: + return new PurviewRequestException(statusCode, endpointName); + } + } + + /// + /// Creates a new instance. + /// + /// The token credential used to authenticate with Purview. + /// The settings used for purview requests. + /// The HttpClient used to make network requests to Purview. + /// The logger used to log information from the middleware. + public PurviewClient(TokenCredential tokenCredential, PurviewSettings purviewSettings, HttpClient httpClient, ILogger logger) + { + this._tokenCredential = tokenCredential; + this._httpClient = httpClient; + + this._scopes = [$"https://{purviewSettings.GraphBaseUri.Host}/.default"]; + this._graphUri = purviewSettings.GraphBaseUri.ToString().TrimEnd('/'); + this._logger = logger ?? NullLogger.Instance; + } + + private static TokenInfo ExtractTokenInfo(string tokenString) + { + // Split JWT and decode payload + string[] parts = tokenString.Split('.'); + if (parts.Length < 2) + { + throw new PurviewRequestException("Invalid JWT access token format."); + } + + string payload = parts[1]; + // Pad base64 string if needed + int mod4 = payload.Length % 4; + if (mod4 > 0) + { + payload += new string('=', 4 - mod4); + } + + byte[] bytes = Convert.FromBase64String(payload.Replace('-', '+').Replace('_', '/')); + string json = Encoding.UTF8.GetString(bytes); + + using var doc = JsonDocument.Parse(json); + var root = doc.RootElement; + + string? objectId = root.TryGetProperty("oid", out var oidProp) ? oidProp.GetString() : null; + string? idType = root.TryGetProperty("idtyp", out var idtypProp) ? idtypProp.GetString() : null; + string? tenant = root.TryGetProperty("tid", out var tidProp) ? tidProp.GetString() : null; + string? clientId = root.TryGetProperty("appid", out var appidProp) ? appidProp.GetString() : null; + + string? userId = idType == "user" ? objectId : null; + + return new TokenInfo + { + UserId = userId, + TenantId = tenant, + ClientId = clientId + }; + } + + /// + public async Task GetUserInfoFromTokenAsync(CancellationToken cancellationToken, string? tenantId = default) + { + TokenRequestContext tokenRequestContext = tenantId == null ? new(this._scopes) : new(this._scopes, tenantId: tenantId); + AccessToken token = await this._tokenCredential.GetTokenAsync(tokenRequestContext, cancellationToken).ConfigureAwait(false); + + string tokenString = token.Token; + + return ExtractTokenInfo(tokenString); + } + + /// + public async Task ProcessContentAsync(ProcessContentRequest request, CancellationToken cancellationToken) + { + var token = await this._tokenCredential.GetTokenAsync(new TokenRequestContext(this._scopes, tenantId: request.TenantId), cancellationToken).ConfigureAwait(false); + string userId = request.UserId; + + string uri = $"{this._graphUri}/users/{userId}/dataSecurityAndGovernance/processContent"; + + using (HttpRequestMessage message = new(HttpMethod.Post, new Uri(uri))) + { + message.Headers.Add("Authorization", $"Bearer {token.Token}"); + message.Headers.Add("User-Agent", "agent-framework-dotnet"); + + if (request.ScopeIdentifier != null) + { + message.Headers.Add("If-None-Match", request.ScopeIdentifier); + } + + string content = JsonSerializer.Serialize(request, PurviewSerializationUtils.SerializationSettings.GetTypeInfo(typeof(ProcessContentRequest))); + message.Content = new StringContent(content, Encoding.UTF8, "application/json"); + + HttpResponseMessage response; + try + { + response = await this._httpClient.SendAsync(message, cancellationToken).ConfigureAwait(false); + } + catch (HttpRequestException e) + { + this._logger.LogError(e, "Http error while processing content."); + throw new PurviewRequestException("Http error occurred while processing content.", e); + } + +#if NET5_0_OR_GREATER + // Pass the cancellation token if that method is available. + string responseContent = await response.Content.ReadAsStringAsync(cancellationToken).ConfigureAwait(false); +#else + string responseContent = await response.Content.ReadAsStringAsync().ConfigureAwait(false); +#endif + + if (response.StatusCode == HttpStatusCode.OK || response.StatusCode == HttpStatusCode.Accepted) + { + ProcessContentResponse? deserializedResponse; + try + { + JsonTypeInfo typeInfo = (JsonTypeInfo)PurviewSerializationUtils.SerializationSettings.GetTypeInfo(typeof(ProcessContentResponse)); + deserializedResponse = JsonSerializer.Deserialize(responseContent, typeInfo); + } + catch (JsonException jsonException) + { + const string DeserializeExceptionError = "Failed to deserialize ProcessContent response."; + this._logger.LogError(jsonException, DeserializeExceptionError); + throw new PurviewRequestException(DeserializeExceptionError, jsonException); + } + + if (deserializedResponse != null) + { + return deserializedResponse; + } + + const string DeserializeError = "Failed to deserialize ProcessContent response. Response was null."; + this._logger.LogError(DeserializeError); + throw new PurviewRequestException(DeserializeError); + } + + if (this._logger.IsEnabled(LogLevel.Error)) + { + this._logger.LogError("Failed to process content. Status code: {StatusCode}", response.StatusCode); + } + + throw CreateExceptionForStatusCode(response.StatusCode, "processContent"); + } + } + + /// + public async Task GetProtectionScopesAsync(ProtectionScopesRequest request, CancellationToken cancellationToken) + { + var token = await this._tokenCredential.GetTokenAsync(new TokenRequestContext(this._scopes), cancellationToken).ConfigureAwait(false); + string userId = request.UserId; + + string uri = $"{this._graphUri}/users/{userId}/dataSecurityAndGovernance/protectionScopes/compute"; + + using (HttpRequestMessage message = new(HttpMethod.Post, new Uri(uri))) + { + message.Headers.Add("Authorization", $"Bearer {token.Token}"); + message.Headers.Add("User-Agent", "agent-framework-dotnet"); + + var typeinfo = PurviewSerializationUtils.SerializationSettings.GetTypeInfo(typeof(ProtectionScopesRequest)); + string content = JsonSerializer.Serialize(request, typeinfo); + message.Content = new StringContent(content, Encoding.UTF8, "application/json"); + + HttpResponseMessage response; + try + { + response = await this._httpClient.SendAsync(message, cancellationToken).ConfigureAwait(false); + } + catch (HttpRequestException e) + { + this._logger.LogError(e, "Http error while retrieving protection scopes."); + throw new PurviewRequestException("Http error occurred while retrieving protection scopes.", e); + } + + if (response.StatusCode == HttpStatusCode.OK) + { +#if NET5_0_OR_GREATER + // Pass the cancellation token if that method is available. + string responseContent = await response.Content.ReadAsStringAsync(cancellationToken).ConfigureAwait(false); +#else + string responseContent = await response.Content.ReadAsStringAsync().ConfigureAwait(false); +#endif + ProtectionScopesResponse? deserializedResponse; + try + { + JsonTypeInfo typeInfo = (JsonTypeInfo)PurviewSerializationUtils.SerializationSettings.GetTypeInfo(typeof(ProtectionScopesResponse)); + deserializedResponse = JsonSerializer.Deserialize(responseContent, typeInfo); + } + catch (JsonException jsonException) + { + const string DeserializeExceptionError = "Failed to deserialize ProtectionScopes response."; + this._logger.LogError(jsonException, DeserializeExceptionError); + throw new PurviewRequestException(DeserializeExceptionError, jsonException); + } + + if (deserializedResponse != null) + { + deserializedResponse.ScopeIdentifier = response.Headers.ETag?.Tag; + return deserializedResponse; + } + + const string DeserializeError = "Failed to deserialize ProtectionScopes response."; + this._logger.LogError(DeserializeError); + throw new PurviewRequestException(DeserializeError); + } + + if (this._logger.IsEnabled(LogLevel.Error)) + { + this._logger.LogError("Failed to retrieve protection scopes. Status code: {StatusCode}", response.StatusCode); + } + + throw CreateExceptionForStatusCode(response.StatusCode, "protectionScopes/compute"); + } + } + + /// + public async Task SendContentActivitiesAsync(ContentActivitiesRequest request, CancellationToken cancellationToken) + { + var token = await this._tokenCredential.GetTokenAsync(new TokenRequestContext(this._scopes), cancellationToken).ConfigureAwait(false); + string userId = request.UserId; + + string uri = $"{this._graphUri}/{userId}/dataSecurityAndGovernance/activities/contentActivities"; + + using (HttpRequestMessage message = new(HttpMethod.Post, new Uri(uri))) + { + message.Headers.Add("Authorization", $"Bearer {token.Token}"); + message.Headers.Add("User-Agent", "agent-framework-dotnet"); + string content = JsonSerializer.Serialize(request, PurviewSerializationUtils.SerializationSettings.GetTypeInfo(typeof(ContentActivitiesRequest))); + message.Content = new StringContent(content, Encoding.UTF8, "application/json"); + HttpResponseMessage response; + + try + { + response = await this._httpClient.SendAsync(message, cancellationToken).ConfigureAwait(false); + } + catch (HttpRequestException e) + { + this._logger.LogError(e, "Http error while creating content activities."); + throw new PurviewRequestException("Http error occurred while creating content activities.", e); + } + + if (response.StatusCode == HttpStatusCode.Created) + { +#if NET5_0_OR_GREATER + // Pass the cancellation token if that method is available. + string responseContent = await response.Content.ReadAsStringAsync(cancellationToken).ConfigureAwait(false); +#else + string responseContent = await response.Content.ReadAsStringAsync().ConfigureAwait(false); +#endif + ContentActivitiesResponse? deserializedResponse; + + try + { + JsonTypeInfo typeInfo = (JsonTypeInfo)PurviewSerializationUtils.SerializationSettings.GetTypeInfo(typeof(ContentActivitiesResponse)); + deserializedResponse = JsonSerializer.Deserialize(responseContent, typeInfo); + } + catch (JsonException jsonException) + { + const string DeserializeExceptionError = "Failed to deserialize ContentActivities response."; + this._logger.LogError(jsonException, DeserializeExceptionError); + throw new PurviewRequestException(DeserializeExceptionError, jsonException); + } + + if (deserializedResponse != null) + { + return deserializedResponse; + } + + const string DeserializeError = "Failed to deserialize ContentActivities response."; + this._logger.LogError(DeserializeError); + throw new PurviewRequestException(DeserializeError); + } + + if (this._logger.IsEnabled(LogLevel.Error)) + { + this._logger.LogError("Failed to create content activities. Status code: {StatusCode}", response.StatusCode); + } + + throw CreateExceptionForStatusCode(response.StatusCode, "contentActivities"); + } + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/PurviewExtensions.cs b/dotnet/src/Microsoft.Agents.AI.Purview/PurviewExtensions.cs new file mode 100644 index 0000000000..4095345d99 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/PurviewExtensions.cs @@ -0,0 +1,118 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Net.Http; +using System.Threading.Channels; +using Azure.Core; +using Microsoft.Agents.AI.Purview.Models.Jobs; +using Microsoft.Extensions.AI; +using Microsoft.Extensions.Caching.Distributed; +using Microsoft.Extensions.Caching.Memory; +using Microsoft.Extensions.DependencyInjection; +using Microsoft.Extensions.Logging; +using Microsoft.Extensions.Logging.Abstractions; +using Microsoft.Extensions.Options; + +namespace Microsoft.Agents.AI.Purview; + +/// +/// Extension methods to add Purview capabilities to an . +/// +public static class PurviewExtensions +{ + private static PurviewWrapper CreateWrapper(TokenCredential tokenCredential, PurviewSettings purviewSettings, ILogger? logger = null, IDistributedCache? cache = null) + { + MemoryDistributedCacheOptions options = new() + { + SizeLimit = purviewSettings.InMemoryCacheSizeLimit, + }; + + IDistributedCache distributedCache = cache ?? new MemoryDistributedCache(Options.Create(options)); + + ServiceCollection services = new(); + services.AddSingleton(tokenCredential); + services.AddSingleton(purviewSettings); + services.AddSingleton(); + services.AddSingleton(); + services.AddSingleton(distributedCache); + services.AddSingleton(); + services.AddSingleton(); + services.AddSingleton(logger ?? NullLogger.Instance); + services.AddSingleton(); + services.AddSingleton(Channel.CreateBounded(purviewSettings.PendingBackgroundJobLimit)); + services.AddSingleton(); + services.AddSingleton(); + ServiceProvider serviceProvider = services.BuildServiceProvider(); + + return serviceProvider.GetRequiredService(); + } + + /// + /// Adds Purview capabilities to an . + /// + /// The AI Agent builder for the . + /// The token credential used to authenticate with Purview. + /// The settings for communication with Purview. + /// The logger to use for logging. + /// The distributed cache to use for caching Purview responses. An in memory cache will be used if this is null. + /// The updated + public static AIAgentBuilder WithPurview(this AIAgentBuilder builder, TokenCredential tokenCredential, PurviewSettings purviewSettings, ILogger? logger = null, IDistributedCache? cache = null) + { + PurviewWrapper purviewWrapper = CreateWrapper(tokenCredential, purviewSettings, logger, cache); + return builder.Use((innerAgent) => new PurviewAgent(innerAgent, purviewWrapper)); + } + + /// + /// Adds Purview capabilities to a . + /// + /// The chat client builder for the . + /// The token credential used to authenticate with Purview. + /// The settings for communication with Purview. + /// The logger to use for logging. + /// The distributed cache to use for caching Purview responses. An in memory cache will be used if this is null. + /// The updated + public static ChatClientBuilder WithPurview(this ChatClientBuilder builder, TokenCredential tokenCredential, PurviewSettings purviewSettings, ILogger? logger = null, IDistributedCache? cache = null) + { + PurviewWrapper purviewWrapper = CreateWrapper(tokenCredential, purviewSettings, logger, cache); + return builder.Use((innerChatClient) => new PurviewChatClient(innerChatClient, purviewWrapper)); + } + + /// + /// Creates a Purview middleware function for use with a . + /// + /// The token credential used to authenticate with Purview. + /// The settings for communication with Purview. + /// The logger to use for logging. + /// The distributed cache to use for caching Purview responses. An in memory cache will be used if this is null. + /// A chat middleware delegate. + public static Func PurviewChatMiddleware(TokenCredential tokenCredential, PurviewSettings purviewSettings, ILogger? logger = null, IDistributedCache? cache = null) + { + PurviewWrapper purviewWrapper = CreateWrapper(tokenCredential, purviewSettings, logger, cache); + return (innerChatClient) => new PurviewChatClient(innerChatClient, purviewWrapper); + } + + /// + /// Creates a Purview middleware function for use with an . + /// + /// The token credential used to authenticate with Purview. + /// The settings for communication with Purview. + /// The logger to use for logging. + /// The distributed cache to use for caching Purview responses. An in memory cache will be used if this is null. + /// An agent middleware delegate. + public static Func PurviewAgentMiddleware(TokenCredential tokenCredential, PurviewSettings purviewSettings, ILogger? logger = null, IDistributedCache? cache = null) + { + PurviewWrapper purviewWrapper = CreateWrapper(tokenCredential, purviewSettings, logger, cache); + return (innerAgent) => new PurviewAgent(innerAgent, purviewWrapper); + } + + /// + /// Sets the user id for a message. + /// + /// The message. + /// The id of the owner of the message. + public static void SetUserId(this ChatMessage message, Guid userId) + { + message.AdditionalProperties ??= []; + message.AdditionalProperties[Constants.UserId] = userId.ToString(); + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/PurviewLocationType.cs b/dotnet/src/Microsoft.Agents.AI.Purview/PurviewLocationType.cs new file mode 100644 index 0000000000..4fcc145f0b --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/PurviewLocationType.cs @@ -0,0 +1,24 @@ +// Copyright (c) Microsoft. All rights reserved. + +namespace Microsoft.Agents.AI.Purview; + +/// +/// The type of location for Purview policy evaluation. +/// +public enum PurviewLocationType +{ + /// + /// An application location. + /// + Application, + + /// + /// A URI location. + /// + Uri, + + /// + /// A domain name location. + /// + Domain +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/PurviewSettings.cs b/dotnet/src/Microsoft.Agents.AI.Purview/PurviewSettings.cs new file mode 100644 index 0000000000..cb400805c6 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/PurviewSettings.cs @@ -0,0 +1,88 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; + +namespace Microsoft.Agents.AI.Purview; + +/// +/// Represents the configuration settings for a Purview application, including tenant information, application name, and +/// optional default user settings. +/// +/// This class is used to encapsulate the necessary configuration details for interacting with Purview +/// services. It includes the tenant ID and application name, which are required, and an optional default user ID that +/// can be used for requests where a specific user ID is not provided. +public class PurviewSettings +{ + /// + /// Initializes a new instance of the class. + /// + /// The publicly visible name of the application. + public PurviewSettings(string appName) + { + this.AppName = appName; + } + + /// + /// The publicly visible app name of the application. + /// + public string AppName { get; set; } + + /// + /// The version string of the application. + /// + public string? AppVersion { get; set; } + + /// + /// The tenant id of the user making the request. + /// If this is not provided, the tenant id will be inferred from the token. + /// + public string? TenantId { get; set; } + + /// + /// Gets or sets the location of the Purview resource. + /// If this is not provided, a location containing the client id will be used instead. + /// + public PurviewAppLocation? PurviewAppLocation { get; set; } + + /// + /// Gets or sets a flag indicating whether to ignore exceptions when processing Purview requests. False by default. + /// If set to true, exceptions calling Purview will be logged but not thrown. + /// + public bool IgnoreExceptions { get; set; } + + /// + /// Gets or sets the base URI for the Microsoft Graph API. + /// Set to graph v1.0 by default. + /// + public Uri GraphBaseUri { get; set; } = new Uri("https://graph.microsoft.com/v1.0/"); + + /// + /// Gets or sets the message to display when a prompt is blocked by Purview policies. + /// + public string BlockedPromptMessage { get; set; } = "Prompt blocked by policies"; + + /// + /// Gets or sets the message to display when a response is blocked by Purview policies. + /// + public string BlockedResponseMessage { get; set; } = "Response blocked by policies"; + + /// + /// The size limit of the default in memory cache in bytes. This only applies if no cache is provided when creating Purview resources. + /// + public long? InMemoryCacheSizeLimit { get; set; } = 100_000_000; + + /// + /// The TTL of each cache entry. + /// + public TimeSpan CacheTTL { get; set; } = TimeSpan.FromMinutes(30); + + /// + /// The maximum number of background jobs that can be queued up. + /// + public int PendingBackgroundJobLimit { get; set; } = 100; + + /// + /// The maximum number of concurrent job consumers. + /// + public int MaxConcurrentJobConsumers { get; set; } = 10; +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/PurviewWrapper.cs b/dotnet/src/Microsoft.Agents.AI.Purview/PurviewWrapper.cs new file mode 100644 index 0000000000..a818fb264f --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/PurviewWrapper.cs @@ -0,0 +1,209 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Collections.Generic; +using System.Threading; +using System.Threading.Tasks; +using Microsoft.Agents.AI.Purview.Models.Common; +using Microsoft.Extensions.AI; +using Microsoft.Extensions.Logging; + +namespace Microsoft.Agents.AI.Purview; + +/// +/// A delegating agent that connects to Microsoft Purview. +/// +internal sealed class PurviewWrapper : IDisposable +{ + private readonly ILogger _logger; + private readonly IScopedContentProcessor _scopedProcessor; + private readonly PurviewSettings _purviewSettings; + private readonly IChannelHandler _channelHandler; + + /// + /// Creates a new instance. + /// + /// The scoped processor used to orchestrate the calls to Purview. + /// The settings for Purview integration. + /// The logger used for logging. + /// The channel handler used to queue background jobs and add job runners. + public PurviewWrapper(IScopedContentProcessor scopedProcessor, PurviewSettings purviewSettings, ILogger logger, IChannelHandler channelHandler) + { + this._scopedProcessor = scopedProcessor; + this._purviewSettings = purviewSettings; + this._logger = logger; + this._channelHandler = channelHandler; + } + + private static string GetThreadIdFromAgentThread(AgentThread? thread, IEnumerable messages) + { + if (thread is ChatClientAgentThread chatClientAgentThread && + chatClientAgentThread.ConversationId != null) + { + return chatClientAgentThread.ConversationId; + } + + foreach (ChatMessage message in messages) + { + if (message.AdditionalProperties != null && + message.AdditionalProperties.TryGetValue(Constants.ConversationId, out object? conversationId) && + conversationId != null) + { + return conversationId.ToString() ?? Guid.NewGuid().ToString(); + } + } + + return Guid.NewGuid().ToString(); + } + + /// + /// Processes a prompt and response exchange at a chat client level. + /// + /// The messages sent to the chat client. + /// The chat options used with the chat client. + /// The wrapped chat client. + /// The cancellation token used to interrupt async operations. + /// The chat client's response. This could be the response from the chat client or a message indicating that Purview has blocked the prompt or response. + public async Task ProcessChatContentAsync(IEnumerable messages, ChatOptions? options, IChatClient innerChatClient, CancellationToken cancellationToken) + { + string? resolvedUserId = null; + + try + { + (bool shouldBlockPrompt, resolvedUserId) = await this._scopedProcessor.ProcessMessagesAsync(messages, options?.ConversationId, Activity.UploadText, this._purviewSettings, null, cancellationToken).ConfigureAwait(false); + if (shouldBlockPrompt) + { + if (this._logger.IsEnabled(LogLevel.Information)) + { + this._logger.LogInformation("Prompt blocked by policy. Sending message: {Message}", this._purviewSettings.BlockedPromptMessage); + } + + return new ChatResponse(new ChatMessage(ChatRole.System, this._purviewSettings.BlockedPromptMessage)); + } + } + catch (Exception ex) + { + if (this._logger.IsEnabled(LogLevel.Error)) + { + this._logger.LogError(ex, "Error processing prompt: {ExceptionMessage}", ex.Message); + } + + if (!this._purviewSettings.IgnoreExceptions) + { + throw; + } + } + + ChatResponse response = await innerChatClient.GetResponseAsync(messages, options, cancellationToken).ConfigureAwait(false); + + try + { + (bool shouldBlockResponse, _) = await this._scopedProcessor.ProcessMessagesAsync(response.Messages, options?.ConversationId, Activity.UploadText, this._purviewSettings, resolvedUserId, cancellationToken).ConfigureAwait(false); + if (shouldBlockResponse) + { + if (this._logger.IsEnabled(LogLevel.Information)) + { + this._logger.LogInformation("Response blocked by policy. Sending message: {Message}", this._purviewSettings.BlockedResponseMessage); + } + + return new ChatResponse(new ChatMessage(ChatRole.System, this._purviewSettings.BlockedResponseMessage)); + } + } + catch (Exception ex) + { + if (this._logger.IsEnabled(LogLevel.Error)) + { + this._logger.LogError(ex, "Error processing response: {ExceptionMessage}", ex.Message); + } + + if (!this._purviewSettings.IgnoreExceptions) + { + throw; + } + } + + return response; + } + + /// + /// Processes a prompt and response exchange at an agent level. + /// + /// The messages sent to the agent. + /// The thread used for this agent conversation. + /// The options used with this agent. + /// The wrapped agent. + /// The cancellation token used to interrupt async operations. + /// The agent's response. This could be the response from the agent or a message indicating that Purview has blocked the prompt or response. + public async Task ProcessAgentContentAsync(IEnumerable messages, AgentThread? thread, AgentRunOptions? options, AIAgent innerAgent, CancellationToken cancellationToken) + { + string threadId = GetThreadIdFromAgentThread(thread, messages); + + string? resolvedUserId = null; + + try + { + (bool shouldBlockPrompt, resolvedUserId) = await this._scopedProcessor.ProcessMessagesAsync(messages, threadId, Activity.UploadText, this._purviewSettings, null, cancellationToken).ConfigureAwait(false); + + if (shouldBlockPrompt) + { + if (this._logger.IsEnabled(LogLevel.Information)) + { + this._logger.LogInformation("Prompt blocked by policy. Sending message: {Message}", this._purviewSettings.BlockedPromptMessage); + } + + return new AgentRunResponse(new ChatMessage(ChatRole.System, this._purviewSettings.BlockedPromptMessage)); + } + } + catch (Exception ex) + { + if (this._logger.IsEnabled(LogLevel.Error)) + { + this._logger.LogError(ex, "Error processing prompt: {ExceptionMessage}", ex.Message); + } + + if (!this._purviewSettings.IgnoreExceptions) + { + throw; + } + } + + AgentRunResponse response = await innerAgent.RunAsync(messages, thread, options, cancellationToken).ConfigureAwait(false); + + try + { + (bool shouldBlockResponse, _) = await this._scopedProcessor.ProcessMessagesAsync(response.Messages, threadId, Activity.UploadText, this._purviewSettings, resolvedUserId, cancellationToken).ConfigureAwait(false); + + if (shouldBlockResponse) + { + if (this._logger.IsEnabled(LogLevel.Information)) + { + this._logger.LogInformation("Response blocked by policy. Sending message: {Message}", this._purviewSettings.BlockedResponseMessage); + } + + return new AgentRunResponse(new ChatMessage(ChatRole.System, this._purviewSettings.BlockedResponseMessage)); + } + } + catch (Exception ex) + { + if (this._logger.IsEnabled(LogLevel.Error)) + { + this._logger.LogError(ex, "Error processing response: {ExceptionMessage}", ex.Message); + } + + if (!this._purviewSettings.IgnoreExceptions) + { + throw; + } + } + + return response; + } + + /// + public void Dispose() + { +#pragma warning disable VSTHRD002 // Need to wait for pending jobs to complete. + this._channelHandler.StopAndWaitForCompletionAsync().GetAwaiter().GetResult(); +#pragma warning restore VSTHRD002 // Need to wait for pending jobs to complete. + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/README.md b/dotnet/src/Microsoft.Agents.AI.Purview/README.md new file mode 100644 index 0000000000..3e46ceff65 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/README.md @@ -0,0 +1,263 @@ +# Microsoft Agent Framework - Purview Integration (Dotnet) + +The Purview plugin for the Microsoft Agent Framework adds Purview policy evaluation to the Microsoft Agent Framework. +It lets you enforce data security and governance policies on both the *prompt* (user input + conversation history) and the *model response* before they proceed further in your workflow. + +> Status: **Preview** + +### Key Features + +- Middleware-based policy enforcement (agent-level and chat-client level) +- Blocks or allows content at both ingress (prompt) and egress (response) +- Works with any `IChatClient` or `AIAgent` using the standard Agent Framework middleware pipeline. +- Authenticates to Purview using `TokenCredential`s +- Simple configuration using `PurviewSettings` +- Configurable caching using `IDistributedCache` +- `WithPurview` Extension methods to easily apply middleware to a `ChatClientBuilder` or `AIAgentBuilder` + +### When to Use +Add Purview when you need to: + +- Prevent sensitive or disallowed content from being sent to an LLM +- Prevent model output containing disallowed data from leaving the system +- Apply centrally managed policies without rewriting agent logic + +--- + + +## Quick Start + +``` csharp +using Azure.AI.OpenAI; +using Azure.Core; +using Azure.Identity; +using Microsoft.Agents.AI; +using Microsoft.Agents.AI.Purview; +using Microsoft.Extensions.AI; + +Uri endpoint = new Uri("..."); // The endpoint of Azure OpenAI instance. +string deploymentName = "..."; // The deployment name of your Azure OpenAI instance ex: gpt-4o-mini +string purviewClientAppId = "..."; // The client id of your entra app registration. + +// This will get a user token for an entra app configured to call the Purview API. +// Any TokenCredential with permissions to call the Purview API can be used here. +TokenCredential browserCredential = new InteractiveBrowserCredential( + new InteractiveBrowserCredentialOptions + { + ClientId = purviewClientAppId + }); + +IChatClient client = new AzureOpenAIClient( + new Uri(endpoint), + new AzureCliCredential()) + .GetOpenAIResponseClient(deploymentName) + .AsIChatClient() + .AsBuilder() + .WithPurview(browserCredential, new PurviewSettings("My Sample App")) + .Build(); + +using (client) +{ + Console.WriteLine("Enter a prompt to send to the client:"); + string? promptText = Console.ReadLine(); + + if (!string.IsNullOrEmpty(promptText)) + { + // Invoke the agent and output the text result. + Console.WriteLine(await client.GetResponseAsync(promptText)); + } +} +``` + +If a policy violation is detected on the prompt, the middleware interrupts the run and outputs the message: `"Prompt blocked by policies"`. If on the response, the result becomes `"Response blocked by policies"`. + +--- + +## Authentication + +The Purview middleware uses Azure.Core TokenCredential objects for authentication. + +The plugin requires the following Graph permissions: +- ProtectionScopes.Compute.All : [userProtectionScopeContainer](https://learn.microsoft.com/en-us/graph/api/userprotectionscopecontainer-compute) +- Content.Process.All : [processContent](https://learn.microsoft.com/en-us/graph/api/userdatasecurityandgovernance-processcontent) +- ContentActivity.Write : [contentActivity](https://learn.microsoft.com/en-us/graph/api/activitiescontainer-post-contentactivities) + +Authentication with user tokens is preferred. If authenticating with app tokens, the agent-framework caller will need to provide an entra user id for each `ChatMessage` send to the agent/client. This user id can be set using the `SetUserId` extension method, or by setting the `"userId"` field of the `AdditionalProperties` dictionary. + +``` csharp +// Manually +var message = new ChatMessage(ChatRole.User, promptText); +if (message.AdditionalProperties == null) +{ + message.AdditionalProperties = new AdditionalPropertiesDictionary(); +} +message.AdditionalProperties["userId"] = ""; + +// Or with the extension method +var message = new ChatMessage(ChatRole.User, promptText); +message.SetUserId(new Guid("")); +``` + +### Tenant Enablement for Purview +- The tenant requires an e5 license and consumptive billing setup. +- [Data Loss Prevention](https://learn.microsoft.com/en-us/purview/dlp-create-deploy-policy) or [Data Collection Policies](https://learn.microsoft.com/en-us/purview/collection-policies-policy-reference) policies that apply to the user are required to enable classification and message ingestion (Process Content API). Otherwise, messages will only be logged in Purview's Audit log (Content Activities API). + +## Configuration + +### Settings + +The Purview middleware can be customized and configured using the `PurviewSettings` class. + +#### `PurviewSettings` + +| Field | Type | Purpose | +| ----- | ---- | ------- | +| AppName | string | The publicly visible app name of the application. | +| AppVersion | string? | (Optional) The version string of the application. | +| TenantId | string? | (Optional) The tenant id of the user making the request. If not provided, this will be inferred from the token. | +| PurviewAppLocation | PurviewAppLocation? | (Optional) The location of the Purview resource used during policy evaluation. If not provided, a location containing the application client id will be used instead. | +| IgnoreExceptions | bool | (Optional, `false` by default) Determines if the exceptions thrown in the Purview middleware should be ignored. If set to true, exceptions will be logged but not thrown. | +| GraphBaseUri | Uri | (Optional, https://graph.microsoft.com/v1.0/ by default) The base URI used for calls to Purview's Microsoft Graph APIs. | +| BlockedPromptMessage | string | (Optional, `"Prompt blocked by policies"` by default) The message returned when a prompt is blocked by Purview. | +| BlockedResponseMessage | string | (Optional, `"Response blocked by policies"` by default) The message returned when a response is blocked by Purview. | +| InMemoryCacheSizeLimit | long? | (Optional, `100_000_000` by default) The size limit of the default in-memory cache in bytes. This only applies if no cache is provided when creating the Purview middleware. | +| CacheTTL | TimeSpan | (Optional, 30 minutes by default) The time to live of each cache entry. | +| PendingBackgroundJobLimit | int | (Optional, 100 by default) The maximum number of pending background jobs that can be queued in the middleware. | +| MaxConcurrentJobConsumers | int | (Optional, 10 by default) The maximum number of concurrent consumers that can run background jobs in the middleware. | + +#### `PurviewAppLocation` + +| Field | Type | Purpose | +| ----- | ---- | ------- | +| LocationType | PurviewLocationType | The type of the location: Application, Uri, Domain. | +| LocationValue | string | The value of the location. | + +#### Location + +The `PurviewAppLocation` field of the `PurviewSettings` object contains the location of the app which is used by Purview for policy evaluation (see [policyLocation](https://learn.microsoft.com/en-us/graph/api/resources/policylocation?view=graph-rest-1.0) for more information). +This location can be set to the URL of the agent app, the domain of the agent app, or the application id of the agent app. + +#### Example + +```csharp +var location = new PurviewAppLocation(PurviewLocationType.Uri, "https://contoso.com/chatagent"); +var settings = new PurviewSettings("My Sample App") +{ + AppVersion = "1.0", + TenantId = "your-tenant-id", + PurviewAppLocation = location, + IgnoreExceptions = false, + GraphBaseUri = new Uri("https://graph.microsoft.com/v1.0/"), + BlockedPromptMessage = "Prompt blocked by policies.", + BlockedResponseMessage = "Response blocked by policies.", + InMemoryCacheSizeLimit = 100_000_000, + CacheTTL = TimeSpan.FromMinutes(30), + PendingBackgroundJobLimit = 100, + MaxConcurrentJobConsumers = 10, +}; + +// ... Set up credential and client builder ... + +var client = builder.WithPurview(credential, settings).Build(); +``` + +#### Customizing Blocked Messages + +This is useful for: +- Providing more user-friendly error messages +- Including support contact information +- Localizing messages for different languages +- Adding branding or specific guidance for your application + +``` csharp +var settings = new PurviewSettings("My Sample App") +{ + BlockedPromptMessage = "Your request contains content that violates our policies. Please rephrase and try again.", + BlockedResponseMessage = "The response was blocked due to policy restrictions. Please contact support if you need assistance.", +}; +``` + +### Selecting Agent vs Chat Middleware + +Use the agent middleware when you already have / want the full agent pipeline: + +``` csharp +AIAgent agent = new AzureOpenAIClient( + new Uri(endpoint), + new AzureCliCredential()) + .GetChatClient(deploymentName) + .CreateAIAgent("You are a helpful assistant.") + .AsBuilder() + .WithPurview(browserCredential, new PurviewSettings("Agent Framework Test App")) + .Build(); +``` + +Use the chat middleware when you attach directly to a chat client (e.g. minimal agent shell or custom orchestration): + +``` csharp +IChatClient client = new AzureOpenAIClient( + new Uri(endpoint), + new AzureCliCredential()) + .GetOpenAIResponseClient(deploymentName) + .AsIChatClient() + .AsBuilder() + .WithPurview(browserCredential, new PurviewSettings("Agent Framework Test App")) + .Build(); +``` + +The policy logic is identical; the only difference is the hook point in the pipeline. + +--- + +## Middleware Lifecycle +1. Before sending the prompt to the agent, the middleware checks the app and user metadata against Purview's protection scopes and evaluates all the `ChatMessage`s in the prompt. +2. If the content was blocked, the middleware returns a `ChatResponse` or `AgentRunResponse` containing the `BlockedPromptMessage` text. The blocked content does not get passed to the agent. +3. If the evaluation did not block the content, the middleware passes the prompt data to the agent and waits for a response. +4. After receiving a response from the agent, the middleware calls Purview again to evaluate the response content. +5. If the content was blocked, the middleware returns a response containing the `BlockedResponseMessage`. + +The user id from the prompt message(s) is reused for the response evaluation so both evaluations map consistently to the same user. + +There are several optimizations to speed up Purview calls. Protection scope lookups (the first step in evaluation) are cached to minimize network calls. +If the policies allow content to be processed offline, the middleware will add the process content request to a channel and run it in a background worker. Similarly, the middleware will run a background request if no scopes apply and the interaction only has to be logged in Audit. + +## Exceptions +| Exception | Scenario | +| --------- | -------- | +| PurviewAuthenticationException | Token acquisition / validation issues | +| PurviewJobException | Errors thrown by a background job | +| PurviewJobLimitExceededException | Errors caused by exceeding the background job limit | +| PurviewPaymentRequiredException | 402 responses from the service | +| PurviewRateLimitException | 429 responses from the service | +| PurviewRequestException | Other errors related to Purview requests | +| PurviewException | Base class for all Purview plugin exceptions | + +Callers' exception handling can be fine-grained + +``` csharp +try +{ + // Code that uses Purview middleware +} +catch (PurviewPaymentRequiredException) +{ + this._logger.LogError("Payment required for Purview."); +} +catch (PurviewAuthenticationException) +{ + this._logger.LogError("Error authenticating to Purview."); +} +``` + +Or broad + +``` csharp +try +{ + // Code that uses Purview middleware +} +catch (PurviewException e) +{ + this._logger.LogError(e, "Purview middleware threw an exception.") +} +``` diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/ScopedContentProcessor.cs b/dotnet/src/Microsoft.Agents.AI.Purview/ScopedContentProcessor.cs new file mode 100644 index 0000000000..454da2d549 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/ScopedContentProcessor.cs @@ -0,0 +1,345 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Collections.Generic; +using System.Threading; +using System.Threading.Tasks; +using Microsoft.Agents.AI.Purview.Models.Common; +using Microsoft.Agents.AI.Purview.Models.Jobs; +using Microsoft.Agents.AI.Purview.Models.Requests; +using Microsoft.Agents.AI.Purview.Models.Responses; +using Microsoft.Extensions.AI; + +namespace Microsoft.Agents.AI.Purview; + +/// +/// Processor class that combines protectionScopes, processContent, and contentActivities calls. +/// +internal sealed class ScopedContentProcessor : IScopedContentProcessor +{ + private readonly IPurviewClient _purviewClient; + private readonly ICacheProvider _cacheProvider; + private readonly IChannelHandler _channelHandler; + + /// + /// Create a new instance of . + /// + /// The purview client to use for purview requests. + /// The cache used to store Purview data. + /// The channel handler used to manage background jobs. + public ScopedContentProcessor(IPurviewClient purviewClient, ICacheProvider cacheProvider, IChannelHandler channelHandler) + { + this._purviewClient = purviewClient; + this._cacheProvider = cacheProvider; + this._channelHandler = channelHandler; + } + + /// + public async Task<(bool shouldBlock, string? userId)> ProcessMessagesAsync(IEnumerable messages, string? threadId, Activity activity, PurviewSettings purviewSettings, string? userId, CancellationToken cancellationToken) + { + List pcRequests = await this.MapMessageToPCRequestsAsync(messages, threadId, activity, purviewSettings, userId, cancellationToken).ConfigureAwait(false); + + bool shouldBlock = false; + string? resolvedUserId = null; + + foreach (ProcessContentRequest pcRequest in pcRequests) + { + resolvedUserId = pcRequest.UserId; + ProcessContentResponse processContentResponse = await this.ProcessContentWithProtectionScopesAsync(pcRequest, cancellationToken).ConfigureAwait(false); + if (processContentResponse.PolicyActions?.Count > 0) + { + foreach (DlpActionInfo policyAction in processContentResponse.PolicyActions) + { + // We need to process all data before blocking, so set the flag and return it outside of this loop. + if (policyAction.Action == DlpAction.BlockAccess) + { + shouldBlock = true; + } + + if (policyAction.RestrictionAction == RestrictionAction.Block) + { + shouldBlock = true; + } + } + } + } + + return (shouldBlock, resolvedUserId); + } + + private static bool TryGetUserIdFromPayload(IEnumerable messages, out string? userId) + { + userId = null; + + foreach (ChatMessage message in messages) + { + if (message.AdditionalProperties != null && + message.AdditionalProperties.TryGetValue(Constants.UserId, out userId) && + !string.IsNullOrEmpty(userId)) + { + return true; + } + else if (Guid.TryParse(message.AuthorName, out Guid _)) + { + userId = message.AuthorName; + return true; + } + } + + return false; + } + + /// + /// Transform a list of ChatMessages into a list of ProcessContentRequests. + /// + /// The messages to transform. + /// The id of the message thread. + /// The activity performed on the content. + /// The settings used for purview integration. + /// The entra id of the user who made the interaction. + /// The cancellation token used to cancel async operations. + /// A list of process content requests. + private async Task> MapMessageToPCRequestsAsync(IEnumerable messages, string? threadId, Activity activity, PurviewSettings settings, string? userId, CancellationToken cancellationToken) + { + List pcRequests = []; + TokenInfo? tokenInfo = null; + + bool needUserId = userId == null && TryGetUserIdFromPayload(messages, out userId); + + // Only get user info if the tenant id is null or if there's no location. + // If location is missing, we will create a new location using the client id. + if (settings.TenantId == null || + settings.PurviewAppLocation == null || + needUserId) + { + tokenInfo = await this._purviewClient.GetUserInfoFromTokenAsync(cancellationToken, settings.TenantId).ConfigureAwait(false); + } + + string tenantId = settings.TenantId ?? tokenInfo?.TenantId ?? throw new PurviewRequestException("No tenant id provided or inferred for Purview request. Please provide a tenant id in PurviewSettings or configure the TokenCredential to authenticate to a tenant."); + + foreach (ChatMessage message in messages) + { + string messageId = message.MessageId ?? Guid.NewGuid().ToString(); + ContentBase content = new PurviewTextContent(message.Text); + ProcessConversationMetadata conversationmetadata = new(content, messageId, false, $"Agent Framework Message {messageId}") + { + CorrelationId = threadId ?? Guid.NewGuid().ToString() + }; + ActivityMetadata activityMetadata = new(activity); + PolicyLocation policyLocation; + + if (settings.PurviewAppLocation != null) + { + policyLocation = settings.PurviewAppLocation.GetPolicyLocation(); + } + else if (tokenInfo?.ClientId != null) + { + policyLocation = new($"{Constants.ODataGraphNamespace}.policyLocationApplication", tokenInfo.ClientId); + } + else + { + throw new PurviewRequestException("No app location provided or inferred for Purview request. Please provide an app location in PurviewSettings or configure the TokenCredential to authenticate to an entra app."); + } + + string appVersion = !string.IsNullOrEmpty(settings.AppVersion) ? settings.AppVersion : "Unknown"; + + ProtectedAppMetadata protectedAppMetadata = new(policyLocation) + { + Name = settings.AppName, + Version = appVersion + }; + IntegratedAppMetadata integratedAppMetadata = new() + { + Name = settings.AppName, + Version = appVersion + }; + + DeviceMetadata deviceMetadata = new() + { + OperatingSystemSpecifications = new() + { + OperatingSystemPlatform = "Unknown", + OperatingSystemVersion = "Unknown" + } + }; + ContentToProcess contentToProcess = new([conversationmetadata], activityMetadata, deviceMetadata, integratedAppMetadata, protectedAppMetadata); + + if (userId == null && + tokenInfo?.UserId != null) + { + userId = tokenInfo.UserId; + } + + if (string.IsNullOrEmpty(userId)) + { + throw new PurviewRequestException("No user id provided or inferred for Purview request. Please provide an Entra user id in each message's AuthorName, set a default Entra user id in PurviewSettings, or configure the TokenCredential to authenticate to an Entra user."); + } + + ProcessContentRequest pcRequest = new(contentToProcess, userId, tenantId); + pcRequests.Add(pcRequest); + } + + return pcRequests; + } + + /// + /// Orchestrates process content and protection scopes calls. + /// + /// The process content request. + /// The cancellation token used to cancel async operations. + /// A process content response. This could be a response from the process content API or a response generated from a content activities call. + private async Task ProcessContentWithProtectionScopesAsync(ProcessContentRequest pcRequest, CancellationToken cancellationToken) + { + ProtectionScopesRequest psRequest = CreateProtectionScopesRequest(pcRequest, pcRequest.UserId, pcRequest.TenantId, pcRequest.CorrelationId); + + ProtectionScopesCacheKey cacheKey = new(psRequest); + + ProtectionScopesResponse? cacheResponse = await this._cacheProvider.GetAsync(cacheKey, cancellationToken).ConfigureAwait(false); + + ProtectionScopesResponse psResponse; + + if (cacheResponse != null) + { + psResponse = cacheResponse; + } + else + { + psResponse = await this._purviewClient.GetProtectionScopesAsync(psRequest, cancellationToken).ConfigureAwait(false); + await this._cacheProvider.SetAsync(cacheKey, psResponse, cancellationToken).ConfigureAwait(false); + } + + pcRequest.ScopeIdentifier = psResponse.ScopeIdentifier; + + (bool shouldProcess, List dlpActions, ExecutionMode executionMode) = CheckApplicableScopes(pcRequest, psResponse); + + if (shouldProcess) + { + if (executionMode == ExecutionMode.EvaluateOffline) + { + this._channelHandler.QueueJob(new ProcessContentJob(pcRequest)); + return new ProcessContentResponse(); + } + + ProcessContentResponse pcResponse = await this._purviewClient.ProcessContentAsync(pcRequest, cancellationToken).ConfigureAwait(false); + + if (pcResponse.ProtectionScopeState == ProtectionScopeState.Modified) + { + await this._cacheProvider.RemoveAsync(cacheKey, cancellationToken).ConfigureAwait(false); + } + + pcResponse = CombinePolicyActions(pcResponse, dlpActions); + return pcResponse; + } + + ContentActivitiesRequest caRequest = new(pcRequest.UserId, pcRequest.TenantId, pcRequest.ContentToProcess, pcRequest.CorrelationId); + this._channelHandler.QueueJob(new ContentActivityJob(caRequest)); + + return new ProcessContentResponse(); + } + + /// + /// Dedupe policy actions received from the service. + /// + /// The process content response which may contain DLP actions. + /// DLP actions returned from protection scopes. + /// The process content response with the protection scopes DLP actions added. + private static ProcessContentResponse CombinePolicyActions(ProcessContentResponse pcResponse, List? actionInfos) + { + if (actionInfos?.Count > 0) + { + pcResponse.PolicyActions = pcResponse.PolicyActions is null ? + actionInfos : + [.. pcResponse.PolicyActions, .. actionInfos]; + } + + return pcResponse; + } + + /// + /// Check if any scopes are applicable to the request. + /// + /// The process content request. + /// The protection scopes response that was returned for the process content request. + /// A bool indicating if the content needs to be processed. A list of applicable actions from the scopes response, and the execution mode for the process content request. + private static (bool shouldProcess, List dlpActions, ExecutionMode executionMode) CheckApplicableScopes(ProcessContentRequest pcRequest, ProtectionScopesResponse psResponse) + { + ProtectionScopeActivities requestActivity = TranslateActivity(pcRequest.ContentToProcess.ActivityMetadata.Activity); + + // The location data type is formatted as microsoft.graph.{locationType} + // Sometimes a '#' gets appended by graph during responses, so for the sake of simplicity, + // Split it by '.' and take the last segment. We'll do a case-insensitive endsWith later. + string[] locationSegments = pcRequest.ContentToProcess.ProtectedAppMetadata.ApplicationLocation.DataType.Split('.'); + string locationType = locationSegments.Length > 0 ? locationSegments[locationSegments.Length - 1] : pcRequest.ContentToProcess.ProtectedAppMetadata.ApplicationLocation.Value; + + string locationValue = pcRequest.ContentToProcess.ProtectedAppMetadata.ApplicationLocation.Value; + List dlpActions = []; + bool shouldProcess = false; + ExecutionMode executionMode = ExecutionMode.EvaluateOffline; + + foreach (var scope in psResponse.Scopes ?? Array.Empty()) + { + bool activityMatch = scope.Activities.HasFlag(requestActivity); + bool locationMatch = false; + + foreach (var location in scope.Locations ?? Array.Empty()) + { + locationMatch = location.DataType.EndsWith(locationType, StringComparison.OrdinalIgnoreCase) && location.Value.Equals(locationValue, StringComparison.OrdinalIgnoreCase); + } + + if (activityMatch && locationMatch) + { + shouldProcess = true; + + if (scope.ExecutionMode == ExecutionMode.EvaluateInline) + { + executionMode = ExecutionMode.EvaluateInline; + } + + if (scope.PolicyActions != null) + { + dlpActions.AddRange(scope.PolicyActions); + } + } + } + + return (shouldProcess, dlpActions, executionMode); + } + + /// + /// Create a ProtectionScopesRequest for the given content ProcessContentRequest. + /// + /// The process content request. + /// The entra user id of the user who sent the data. + /// The tenant id of the user who sent the data. + /// The correlation id of the request. + /// The protection scopes request generated from the process content request. + private static ProtectionScopesRequest CreateProtectionScopesRequest(ProcessContentRequest pcRequest, string userId, string tenantId, Guid correlationId) + { + return new ProtectionScopesRequest(userId, tenantId) + { + Activities = TranslateActivity(pcRequest.ContentToProcess.ActivityMetadata.Activity), + Locations = [pcRequest.ContentToProcess.ProtectedAppMetadata.ApplicationLocation], + DeviceMetadata = pcRequest.ContentToProcess.DeviceMetadata, + IntegratedAppMetadata = pcRequest.ContentToProcess.IntegratedAppMetadata, + CorrelationId = correlationId + }; + } + + /// + /// Map process content activity to protection scope activity. + /// + /// The process content activity. + /// The protection scopes activity. + private static ProtectionScopeActivities TranslateActivity(Activity activity) + { + return activity switch + { + Activity.Unknown => ProtectionScopeActivities.None, + Activity.UploadText => ProtectionScopeActivities.UploadText, + Activity.UploadFile => ProtectionScopeActivities.UploadFile, + Activity.DownloadText => ProtectionScopeActivities.DownloadText, + Activity.DownloadFile => ProtectionScopeActivities.DownloadFile, + _ => ProtectionScopeActivities.UnknownFutureValue, + }; + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Purview/Serialization/PurviewSerializationUtils.cs b/dotnet/src/Microsoft.Agents.AI.Purview/Serialization/PurviewSerializationUtils.cs new file mode 100644 index 0000000000..320fbcd3b6 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Purview/Serialization/PurviewSerializationUtils.cs @@ -0,0 +1,41 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json; +using System.Text.Json.Serialization; +using Microsoft.Agents.AI.Purview.Models.Common; +using Microsoft.Agents.AI.Purview.Models.Requests; +using Microsoft.Agents.AI.Purview.Models.Responses; + +namespace Microsoft.Agents.AI.Purview.Serialization; + +/// +/// Source generation context for Purview serialization. +/// +[JsonSerializable(typeof(ProtectionScopesRequest))] +[JsonSerializable(typeof(ProtectionScopesResponse))] +[JsonSerializable(typeof(ProcessContentRequest))] +[JsonSerializable(typeof(ProcessContentResponse))] +[JsonSerializable(typeof(ContentActivitiesRequest))] +[JsonSerializable(typeof(ContentActivitiesResponse))] +[JsonSerializable(typeof(ProtectionScopesCacheKey))] +internal sealed partial class SourceGenerationContext : JsonSerializerContext; + +/// +/// Utility class for Purview serialization settings. +/// +internal static class PurviewSerializationUtils +{ + /// + /// Serialization settings for Purview. + /// + public static JsonSerializerOptions SerializationSettings { get; } = new JsonSerializerOptions + { + PropertyNamingPolicy = JsonNamingPolicy.CamelCase, + PropertyNameCaseInsensitive = true, + WriteIndented = false, + AllowTrailingCommas = false, + DictionaryKeyPolicy = JsonNamingPolicy.CamelCase, + DefaultIgnoreCondition = JsonIgnoreCondition.WhenWritingNull, + TypeInfoResolver = SourceGenerationContext.Default, + }; +} diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative.AzureAI/AzureAgentProvider.cs b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative.AzureAI/AzureAgentProvider.cs new file mode 100644 index 0000000000..c4a613901c --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative.AzureAI/AzureAgentProvider.cs @@ -0,0 +1,264 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.ClientModel.Primitives; +using System.Collections.Generic; +using System.Collections.ObjectModel; +using System.Linq; +using System.Net.Http; +using System.Runtime.CompilerServices; +using System.Text.Json.Nodes; +using System.Threading; +using System.Threading.Tasks; +using Azure.AI.Projects; +using Azure.AI.Projects.OpenAI; +using Azure.Core; +using Microsoft.Extensions.AI; +using OpenAI.Responses; + +namespace Microsoft.Agents.AI.Workflows.Declarative; + +/// +/// Provides functionality to interact with Foundry agents within a specified project context. +/// +/// This class is used to retrieve and manage AI agents associated with a Foundry project. It requires a +/// project endpoint and credentials to authenticate requests. +/// A instance representing the endpoint URL of the Foundry project. This must be a valid, non-null URI pointing to the project. +/// The credentials used to authenticate with the Foundry project. This must be a valid instance of . +public sealed class AzureAgentProvider(Uri projectEndpoint, TokenCredential projectCredentials) : WorkflowAgentProvider +{ + private readonly Dictionary _versionCache = []; + private readonly Dictionary _agentCache = []; + + private AIProjectClient? _agentClient; + private ProjectConversationsClient? _conversationClient; + + /// + /// Optional options used when creating the . + /// + public AIProjectClientOptions? AIProjectClientOptions { get; init; } + + /// + /// Optional options used when invoking the . + /// + public ProjectOpenAIClientOptions? OpenAIClientOptions { get; init; } + + /// + /// An optional instance to be used for making HTTP requests. + /// If not provided, a default client will be used. + /// + public HttpClient? HttpClient { get; init; } + + /// + public override async Task CreateConversationAsync(CancellationToken cancellationToken = default) + { + ProjectConversation conversation = + await this.GetConversationClient() + .CreateProjectConversationAsync(options: null, cancellationToken).ConfigureAwait(false); + + return conversation.Id; + } + + /// + public override async Task CreateMessageAsync(string conversationId, ChatMessage conversationMessage, CancellationToken cancellationToken = default) + { + ReadOnlyCollection newItems = + await this.GetConversationClient().CreateProjectConversationItemsAsync( + conversationId, + items: GetResponseItems(), + include: null, + cancellationToken).ConfigureAwait(false); + + return newItems.AsChatMessages().Single(); + + IEnumerable GetResponseItems() + { + IEnumerable messages = [conversationMessage]; + + foreach (ResponseItem item in messages.AsOpenAIResponseItems()) + { + if (string.IsNullOrEmpty(item.Id)) + { + yield return item; + } + else + { + yield return new ReferenceResponseItem(item.Id); + } + } + } + } + + /// + public override async IAsyncEnumerable InvokeAgentAsync( + string agentId, + string? agentVersion, + string? conversationId, + IEnumerable? messages, + IDictionary? inputArguments, + [EnumeratorCancellation] CancellationToken cancellationToken = default) + { + AgentVersion agentVersionResult = await this.QueryAgentAsync(agentId, agentVersion, cancellationToken).ConfigureAwait(false); + AIAgent agent = await this.GetAgentAsync(agentVersionResult, cancellationToken).ConfigureAwait(false); + + ChatOptions chatOptions = + new() + { + ConversationId = conversationId, + AllowMultipleToolCalls = this.AllowMultipleToolCalls, + }; + + if (inputArguments is not null) + { + JsonNode jsonNode = ConvertDictionaryToJson(inputArguments); + ResponseCreationOptions responseCreationOptions = new(); +#pragma warning disable SCME0001 // Type is for evaluation purposes only and is subject to change or removal in future updates. Suppress this diagnostic to proceed. + responseCreationOptions.Patch.Set("$.structured_inputs"u8, BinaryData.FromString(jsonNode.ToJsonString())); +#pragma warning restore SCME0001 // Type is for evaluation purposes only and is subject to change or removal in future updates. Suppress this diagnostic to proceed. + chatOptions.RawRepresentationFactory = (_) => responseCreationOptions; + } + + ChatClientAgentRunOptions runOptions = new(chatOptions); + + IAsyncEnumerable agentResponse = + messages is not null ? + agent.RunStreamingAsync([.. messages], null, runOptions, cancellationToken) : + agent.RunStreamingAsync([new ChatMessage(ChatRole.User, string.Empty)], null, runOptions, cancellationToken); + + await foreach (AgentRunResponseUpdate update in agentResponse.ConfigureAwait(false)) + { + update.AuthorName = agentVersionResult.Name; + yield return update; + } + } + + private async Task QueryAgentAsync(string agentName, string? agentVersion, CancellationToken cancellationToken = default) + { + string agentKey = $"{agentName}:{agentVersion}"; + if (this._versionCache.TryGetValue(agentKey, out AgentVersion? targetAgent)) + { + return targetAgent; + } + + AIProjectClient client = this.GetAgentClient(); + + if (string.IsNullOrEmpty(agentVersion)) + { + AgentRecord agentRecord = + await client.Agents.GetAgentAsync( + agentName, + cancellationToken).ConfigureAwait(false); + + targetAgent = agentRecord.Versions.Latest; + } + else + { + targetAgent = + await client.Agents.GetAgentVersionAsync( + agentName, + agentVersion, + cancellationToken).ConfigureAwait(false); + } + + this._versionCache[agentKey] = targetAgent; + + return targetAgent; + } + + private async Task GetAgentAsync(AgentVersion agentVersion, CancellationToken cancellationToken = default) + { + if (this._agentCache.TryGetValue(agentVersion.Id, out AIAgent? agent)) + { + return agent; + } + + AIProjectClient client = this.GetAgentClient(); + + agent = client.GetAIAgent(agentVersion, tools: null, clientFactory: null, services: null); + + FunctionInvokingChatClient? functionInvokingClient = agent.GetService(); + if (functionInvokingClient is not null) + { + // Allow concurrent invocations if configured + functionInvokingClient.AllowConcurrentInvocation = this.AllowConcurrentInvocation; + // Allows the caller to respond with function responses + functionInvokingClient.TerminateOnUnknownCalls = true; + // Make functions available for execution. Doesn't change what tool is available for any given agent. + if (this.Functions is not null) + { + if (functionInvokingClient.AdditionalTools is null) + { + functionInvokingClient.AdditionalTools = [.. this.Functions]; + } + else + { + functionInvokingClient.AdditionalTools = [.. functionInvokingClient.AdditionalTools, .. this.Functions]; + } + } + } + + this._agentCache[agentVersion.Id] = agent; + + return agent; + } + + /// + public override async Task GetMessageAsync(string conversationId, string messageId, CancellationToken cancellationToken = default) + { + AgentResponseItem responseItem = await this.GetConversationClient().GetProjectConversationItemAsync(conversationId, messageId, include: null, cancellationToken).ConfigureAwait(false); + ResponseItem[] items = [responseItem.AsOpenAIResponseItem()]; + return items.AsChatMessages().Single(); + } + + /// + public override async IAsyncEnumerable GetMessagesAsync( + string conversationId, + int? limit = null, + string? after = null, + string? before = null, + bool newestFirst = false, + [EnumeratorCancellation] CancellationToken cancellationToken = default) + { + AgentListOrder order = newestFirst ? AgentListOrder.Ascending : AgentListOrder.Descending; + + await foreach (AgentResponseItem responseItem in this.GetConversationClient().GetProjectConversationItemsAsync(conversationId, null, limit, order.ToString(), after, before, include: null, cancellationToken).ConfigureAwait(false)) + { + ResponseItem[] items = [responseItem.AsOpenAIResponseItem()]; + foreach (ChatMessage message in items.AsChatMessages()) + { + yield return message; + } + } + } + + private AIProjectClient GetAgentClient() + { + if (this._agentClient is null) + { + AIProjectClientOptions clientOptions = this.AIProjectClientOptions ?? new(); + + if (this.HttpClient is not null) + { + clientOptions.Transport = new HttpClientPipelineTransport(this.HttpClient); + } + + AIProjectClient newClient = new(projectEndpoint, projectCredentials, clientOptions); + + Interlocked.CompareExchange(ref this._agentClient, newClient, null); + } + + return this._agentClient; + } + + private ProjectConversationsClient GetConversationClient() + { + if (this._conversationClient is null) + { + ProjectConversationsClient conversationClient = this.GetAgentClient().GetProjectOpenAIClient().GetProjectConversationsClient(); + + Interlocked.CompareExchange(ref this._conversationClient, conversationClient, null); + } + + return this._conversationClient; + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative.AzureAI/Microsoft.Agents.AI.Workflows.Declarative.AzureAI.csproj b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative.AzureAI/Microsoft.Agents.AI.Workflows.Declarative.AzureAI.csproj new file mode 100644 index 0000000000..1370b6fdca --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative.AzureAI/Microsoft.Agents.AI.Workflows.Declarative.AzureAI.csproj @@ -0,0 +1,39 @@ + + + + preview + $(NoWarn);MEAI001;OPENAI001 + + + + true + true + true + + + + + + + Microsoft Agent Framework Declarative Workflows Azure AI + Provides Microsoft Agent Framework support for declarative workflows for Azure AI Agents. + + + + + + + + + + + + + + + + + + + + diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/AzureAgentProvider.cs b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/AzureAgentProvider.cs deleted file mode 100644 index ac44890c1c..0000000000 --- a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/AzureAgentProvider.cs +++ /dev/null @@ -1,214 +0,0 @@ -// Copyright (c) Microsoft. All rights reserved. - -using System.Collections.Generic; -using System.Linq; -using System.Net.Http; -using System.Runtime.CompilerServices; -using System.Threading; -using System.Threading.Tasks; -using Azure.AI.Agents.Persistent; -using Azure.Core; -using Azure.Core.Pipeline; -using Microsoft.Extensions.AI; - -namespace Microsoft.Agents.AI.Workflows.Declarative; - -/// -/// Provides functionality to interact with Foundry agents within a specified project context. -/// -/// This class is used to retrieve and manage AI agents associated with a Foundry project. It requires a -/// project endpoint and credentials to authenticate requests. -/// The endpoint URL of the Foundry project. This must be a valid, non-null URI pointing to the project. -/// The credentials used to authenticate with the Foundry project. This must be a valid instance of . -/// An optional instance to be used for making HTTP requests. If not provided, a default client will be used. -public sealed class AzureAgentProvider(string projectEndpoint, TokenCredential projectCredentials, HttpClient? httpClient = null) : WorkflowAgentProvider -{ - private static readonly Dictionary s_roleMap = - new() - { - [ChatRole.User.Value.ToUpperInvariant()] = MessageRole.User, - [ChatRole.Assistant.Value.ToUpperInvariant()] = MessageRole.Agent, - [ChatRole.System.Value.ToUpperInvariant()] = new MessageRole(ChatRole.System.Value), - [ChatRole.Tool.Value.ToUpperInvariant()] = new MessageRole(ChatRole.Tool.Value), - }; - - private PersistentAgentsClient? _agentsClient; - - /// - public override async Task CreateConversationAsync(CancellationToken cancellationToken = default) - { - PersistentAgentThread conversation = - await this.GetAgentsClient().Threads.CreateThreadAsync( - messages: null, - toolResources: null, - metadata: null, - cancellationToken).ConfigureAwait(false); - - return conversation.Id; - } - - /// - public override async Task CreateMessageAsync(string conversationId, ChatMessage conversationMessage, CancellationToken cancellationToken = default) - { - PersistentThreadMessage newMessage = - await this.GetAgentsClient().Messages.CreateMessageAsync( - conversationId, - role: s_roleMap[conversationMessage.Role.Value.ToUpperInvariant()], - contentBlocks: GetContent(), - attachments: null, - metadata: GetMetadata(), - cancellationToken).ConfigureAwait(false); - - return ToChatMessage(newMessage); - - Dictionary? GetMetadata() - { - if (conversationMessage.AdditionalProperties is null) - { - return null; - } - - return conversationMessage.AdditionalProperties.ToDictionary(prop => prop.Key, prop => prop.Value?.ToString() ?? string.Empty); - } - - IEnumerable GetContent() - { - foreach (AIContent content in conversationMessage.Contents) - { - MessageInputContentBlock? contentBlock = - content switch - { - TextContent textContent => new MessageInputTextBlock(textContent.Text), - HostedFileContent fileContent => new MessageInputImageFileBlock(new MessageImageFileParam(fileContent.FileId)), - UriContent uriContent when uriContent.Uri is not null => new MessageInputImageUriBlock(new MessageImageUriParam(uriContent.Uri.ToString())), - DataContent dataContent when dataContent.Uri is not null => new MessageInputImageUriBlock(new MessageImageUriParam(dataContent.Uri)), - _ => null // Unsupported content type - }; - - if (contentBlock is not null) - { - yield return contentBlock; - } - } - } - } - - /// - public override async Task GetAgentAsync(string agentId, CancellationToken cancellationToken = default) - { - ChatClientAgent agent = - await this.GetAgentsClient().GetAIAgentAsync( - agentId, - new ChatOptions() - { - AllowMultipleToolCalls = this.AllowMultipleToolCalls, - }, - clientFactory: null, - cancellationToken).ConfigureAwait(false); - - FunctionInvokingChatClient? functionInvokingClient = agent.GetService(); - if (functionInvokingClient is not null) - { - // Allow concurrent invocations if configured - functionInvokingClient.AllowConcurrentInvocation = this.AllowConcurrentInvocation; - // Allows the caller to respond with function responses - functionInvokingClient.TerminateOnUnknownCalls = true; - // Make functions available for execution. Doesn't change what tool is available for any given agent. - if (this.Functions is not null) - { - if (functionInvokingClient.AdditionalTools is null) - { - functionInvokingClient.AdditionalTools = [.. this.Functions]; - } - else - { - functionInvokingClient.AdditionalTools = [.. functionInvokingClient.AdditionalTools, .. this.Functions]; - } - } - } - - return agent; - } - - /// - public override async Task GetMessageAsync(string conversationId, string messageId, CancellationToken cancellationToken = default) - { - PersistentThreadMessage message = await this.GetAgentsClient().Messages.GetMessageAsync(conversationId, messageId, cancellationToken).ConfigureAwait(false); - return ToChatMessage(message); - } - - /// - public override async IAsyncEnumerable GetMessagesAsync( - string conversationId, - int? limit = null, - string? after = null, - string? before = null, - bool newestFirst = false, - [EnumeratorCancellation] CancellationToken cancellationToken = default) - { - ListSortOrder order = newestFirst ? ListSortOrder.Ascending : ListSortOrder.Descending; - await foreach (PersistentThreadMessage message in this.GetAgentsClient().Messages.GetMessagesAsync(conversationId, runId: null, limit, order, after, before, cancellationToken).ConfigureAwait(false)) - { - yield return ToChatMessage(message); - } - } - - private PersistentAgentsClient GetAgentsClient() - { - if (this._agentsClient is null) - { - PersistentAgentsAdministrationClientOptions clientOptions = new(); - - if (httpClient is not null) - { - clientOptions.Transport = new HttpClientTransport(httpClient); - } - - PersistentAgentsClient newClient = new(projectEndpoint, projectCredentials, clientOptions); - - Interlocked.CompareExchange(ref this._agentsClient, newClient, null); - } - - return this._agentsClient; - } - - private static ChatMessage ToChatMessage(PersistentThreadMessage message) - { - return - new ChatMessage(new ChatRole(message.Role.ToString()), [.. GetContent()]) - { - MessageId = message.Id, - CreatedAt = message.CreatedAt, - AdditionalProperties = GetMetadata() - }; - - IEnumerable GetContent() - { - foreach (MessageContent contentItem in message.ContentItems) - { - AIContent? content = - contentItem switch - { - MessageTextContent textContent => new TextContent(textContent.Text), - MessageImageFileContent imageContent => new HostedFileContent(imageContent.FileId), - _ => null // Unsupported content type - }; - - if (content is not null) - { - yield return content; - } - } - } - - AdditionalPropertiesDictionary? GetMetadata() - { - if (message.Metadata is null) - { - return null; - } - - return new AdditionalPropertiesDictionary(message.Metadata.Select(m => new KeyValuePair(m.Key, m.Value))); - } - } -} diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/CodeGen/CodeTemplate.cs b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/CodeGen/CodeTemplate.cs index 87d9ab748b..af201deb4f 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/CodeGen/CodeTemplate.cs +++ b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/CodeGen/CodeTemplate.cs @@ -13,9 +13,6 @@ namespace Microsoft.Agents.AI.Workflows.Declarative.CodeGen; internal abstract class CodeTemplate { - private StringBuilder? _generationEnvironmentField; - private CompilerErrorCollection? _errorsField; - private List? _indentLengthsField; private bool _endsWithNewline; private string CurrentIndentField { get; set; } = string.Empty; @@ -146,22 +143,19 @@ public StringBuilder GenerationEnvironment { get { - return this._generationEnvironmentField ??= new StringBuilder(); - } - set - { - this._generationEnvironmentField = value; + return field ??= new StringBuilder(); } + set; } /// /// The error collection for the generation process /// - public CompilerErrorCollection Errors => this._errorsField ??= []; + public CompilerErrorCollection Errors => field ??= []; /// /// A list of the lengths of each indent that was added with PushIndent /// - private List indentLengths => this._indentLengthsField ??= []; + private List IndentLengths { get => field ??= []; } /// /// Gets the current indent we use when adding lines to the output @@ -288,7 +282,7 @@ public void PushIndent(string indent) throw new ArgumentNullException(nameof(indent)); } this.CurrentIndentField += indent; - this.indentLengths.Add(indent.Length); + this.IndentLengths.Add(indent.Length); } /// @@ -297,10 +291,10 @@ public void PushIndent(string indent) public string PopIndent() { string returnValue = string.Empty; - if (this.indentLengths.Count > 0) + if (this.IndentLengths.Count > 0) { - int indentLength = this.indentLengths[this.indentLengths.Count - 1]; - this.indentLengths.RemoveAt(this.indentLengths.Count - 1); + int indentLength = this.IndentLengths[this.IndentLengths.Count - 1]; + this.IndentLengths.RemoveAt(this.IndentLengths.Count - 1); if (indentLength > 0) { returnValue = this.CurrentIndentField.Substring(this.CurrentIndentField.Length - indentLength); @@ -315,7 +309,7 @@ public string PopIndent() /// public void ClearIndent() { - this.indentLengths.Clear(); + this.IndentLengths.Clear(); this.CurrentIndentField = string.Empty; } diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/CodeGen/InvokeAzureAgentTemplate.cs b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/CodeGen/InvokeAzureAgentTemplate.cs index 2fe387a5f6..3704d38b21 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/CodeGen/InvokeAzureAgentTemplate.cs +++ b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/CodeGen/InvokeAzureAgentTemplate.cs @@ -61,7 +61,6 @@ public override string TransformText() EvaluateStringExpression(this.Model.ConversationId, "conversationId", isNullable: true); EvaluateBoolExpression(this.Model.Output?.AutoSend, "autoSend", defaultValue: true); - EvaluateMessageTemplate(this.Model.Input?.AdditionalInstructions, "additionalInstructions"); EvaluateListExpression(this.Model.Input?.Messages, "inputMessages"); this.Write(@" @@ -71,7 +70,6 @@ await InvokeAgentAsync( agentName, conversationId, autoSend, - additionalInstructions, inputMessages, cancellationToken).ConfigureAwait(false); diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/CodeGen/InvokeAzureAgentTemplate.tt b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/CodeGen/InvokeAzureAgentTemplate.tt index b46e88588a..48c4acb859 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/CodeGen/InvokeAzureAgentTemplate.tt +++ b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/CodeGen/InvokeAzureAgentTemplate.tt @@ -19,7 +19,6 @@ internal sealed class <#= this.Name #>Executor(FormulaSession session, WorkflowA <# EvaluateStringExpression(this.Model.ConversationId, "conversationId", isNullable: true); EvaluateBoolExpression(this.Model.Output?.AutoSend, "autoSend", defaultValue: true); - EvaluateMessageTemplate(this.Model.Input?.AdditionalInstructions, "additionalInstructions"); EvaluateListExpression(this.Model.Input?.Messages, "inputMessages");#> AgentRunResponse agentResponse = @@ -28,7 +27,6 @@ internal sealed class <#= this.Name #>Executor(FormulaSession session, WorkflowA agentName, conversationId, autoSend, - additionalInstructions, inputMessages, cancellationToken).ConfigureAwait(false); diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Events/AgentFunctionToolRequest.cs b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Events/AgentFunctionToolRequest.cs deleted file mode 100644 index 5f9f878e79..0000000000 --- a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Events/AgentFunctionToolRequest.cs +++ /dev/null @@ -1,30 +0,0 @@ -// Copyright (c) Microsoft. All rights reserved. - -using System.Collections.Generic; -using System.Text.Json.Serialization; -using Microsoft.Extensions.AI; - -namespace Microsoft.Agents.AI.Workflows.Declarative.Events; - -/// -/// Represents one or more function tool requests. -/// -public sealed class AgentFunctionToolRequest -{ - /// - /// The name of the agent associated with the tool request. - /// - public string AgentName { get; } - - /// - /// A list of function tool requests. - /// - public IList FunctionCalls { get; } - - [JsonConstructor] - internal AgentFunctionToolRequest(string agentName, IList functionCalls) - { - this.AgentName = agentName; - this.FunctionCalls = functionCalls; - } -} diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Events/AgentFunctionToolResponse.cs b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Events/AgentFunctionToolResponse.cs deleted file mode 100644 index e03414bd2e..0000000000 --- a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Events/AgentFunctionToolResponse.cs +++ /dev/null @@ -1,52 +0,0 @@ -// Copyright (c) Microsoft. All rights reserved. - -using System.Collections.Generic; -using System.Linq; -using System.Text.Json.Serialization; -using Microsoft.Extensions.AI; - -namespace Microsoft.Agents.AI.Workflows.Declarative.Events; - -/// -/// Represents one or more function tool responses. -/// -public sealed class AgentFunctionToolResponse -{ - /// - /// The name of the agent associated with the tool response. - /// - public string AgentName { get; } - - /// - /// A list of tool responses. - /// - public IList FunctionResults { get; } - - [JsonConstructor] - internal AgentFunctionToolResponse(string agentName, IList functionResults) - { - this.AgentName = agentName; - this.FunctionResults = functionResults; - } - - /// - /// Factory method to create an from an - /// Ensures that all function calls in the request have a corresponding result. - /// - /// The tool request. - /// One or more function results - /// An that can be provided to the workflow. - /// Not all have a corresponding . - public static AgentFunctionToolResponse Create(AgentFunctionToolRequest toolRequest, params IEnumerable functionResults) - { - HashSet callIds = [.. toolRequest.FunctionCalls.Select(call => call.CallId)]; - HashSet resultIds = [.. functionResults.Select(call => call.CallId)]; - - if (!callIds.SetEquals(resultIds)) - { - throw new DeclarativeActionException($"Missing results for: {string.Join(",", callIds.Except(resultIds))}"); - } - - return new AgentFunctionToolResponse(toolRequest.AgentName, [.. functionResults]); - } -} diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Events/AnswerRequest.cs b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Events/AnswerRequest.cs deleted file mode 100644 index 845696a180..0000000000 --- a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Events/AnswerRequest.cs +++ /dev/null @@ -1,27 +0,0 @@ -// Copyright (c) Microsoft. All rights reserved. - -using System.Text.Json.Serialization; - -namespace Microsoft.Agents.AI.Workflows.Declarative.Events; - -/// -/// Represents a request for user input in response to a `Question` action. -/// -public sealed class AnswerRequest -{ - /// - /// An optional prompt for the user. - /// - /// - /// This prompt is utilized for the "Question" action type in the Declarative Workflow, - /// but is redundant when the user is responding to an agent since the agent's message - /// is the implicit prompt. - /// - public string? Prompt { get; } - - [JsonConstructor] - internal AnswerRequest(string? prompt = null) - { - this.Prompt = prompt; - } -} diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Events/AnswerResponse.cs b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Events/AnswerResponse.cs deleted file mode 100644 index 00903f43f0..0000000000 --- a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Events/AnswerResponse.cs +++ /dev/null @@ -1,36 +0,0 @@ -// Copyright (c) Microsoft. All rights reserved. - -using System.Text.Json.Serialization; -using Microsoft.Extensions.AI; - -namespace Microsoft.Agents.AI.Workflows.Declarative.Events; - -/// -/// Represents a user input response. -/// -public sealed class AnswerResponse -{ - /// - /// The response value. - /// - public ChatMessage Value { get; } - - /// - /// Initializes a new instance of the class. - /// - /// The response value. - [JsonConstructor] - public AnswerResponse(ChatMessage value) - { - this.Value = value; - } - - /// - /// Initializes a new instance of the class. - /// - /// The response value. - public AnswerResponse(string value) - { - this.Value = new ChatMessage(ChatRole.User, value); - } -} diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Events/ExternalInputRequest.cs b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Events/ExternalInputRequest.cs new file mode 100644 index 0000000000..8caf374b70 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Events/ExternalInputRequest.cs @@ -0,0 +1,33 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json.Serialization; +using Microsoft.Extensions.AI; + +namespace Microsoft.Agents.AI.Workflows.Declarative.Events; + +/// +/// Represents a request for external input. +/// +public sealed class ExternalInputRequest +{ + /// + /// The source message that triggered the request for external input. + /// + public AgentRunResponse AgentResponse { get; } + + [JsonConstructor] + internal ExternalInputRequest(AgentRunResponse agentResponse) + { + this.AgentResponse = agentResponse; + } + + internal ExternalInputRequest(ChatMessage message) + { + this.AgentResponse = new AgentRunResponse(message); + } + + internal ExternalInputRequest(string text) + { + this.AgentResponse = new AgentRunResponse(new ChatMessage(ChatRole.User, text)); + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Events/ExternalInputResponse.cs b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Events/ExternalInputResponse.cs new file mode 100644 index 0000000000..0653a12ce5 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Events/ExternalInputResponse.cs @@ -0,0 +1,39 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Collections.Generic; +using System.Text.Json.Serialization; +using Microsoft.Extensions.AI; + +namespace Microsoft.Agents.AI.Workflows.Declarative.Events; + +/// +/// Represents the response to a . +/// +public sealed class ExternalInputResponse +{ + /// + /// The message being provided as external input to the workflow. + /// + public IList Messages { get; } + + internal bool HasMessages => this.Messages?.Count > 0; + + /// + /// Initializes a new instance of . + /// + /// The external input message being provided to the workflow. + public ExternalInputResponse(ChatMessage message) + { + this.Messages = [message]; + } + + /// + /// Initializes a new instance of . + /// + /// The external input messages being provided to the workflow. + [JsonConstructor] + public ExternalInputResponse(IList messages) + { + this.Messages = messages; + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Events/UserInputRequest.cs b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Events/UserInputRequest.cs deleted file mode 100644 index 1426025d74..0000000000 --- a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Events/UserInputRequest.cs +++ /dev/null @@ -1,30 +0,0 @@ -// Copyright (c) Microsoft. All rights reserved. - -using System.Collections.Generic; -using System.Text.Json.Serialization; -using Microsoft.Extensions.AI; - -namespace Microsoft.Agents.AI.Workflows.Declarative.Events; - -/// -/// Represents one or more user-input requests. -/// -public sealed class UserInputRequest -{ - /// - /// The name of the agent associated with the tool request. - /// - public string AgentName { get; } - - /// - /// A list of user input requests. - /// - public IList InputRequests { get; } - - [JsonConstructor] - internal UserInputRequest(string agentName, IList inputRequests) - { - this.AgentName = agentName; - this.InputRequests = inputRequests; - } -} diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Events/UserInputResponse.cs b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Events/UserInputResponse.cs deleted file mode 100644 index edb9f3b7cc..0000000000 --- a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Events/UserInputResponse.cs +++ /dev/null @@ -1,52 +0,0 @@ -// Copyright (c) Microsoft. All rights reserved. - -using System.Collections.Generic; -using System.Linq; -using System.Text.Json.Serialization; -using Microsoft.Extensions.AI; - -namespace Microsoft.Agents.AI.Workflows.Declarative.Events; - -/// -/// Represents one or more user-input responses. -/// -public sealed class UserInputResponse -{ - /// - /// The name of the agent associated with the tool request. - /// - public string AgentName { get; } - - /// - /// A list of approval responses. - /// - public IList InputResponses { get; } - - [JsonConstructor] - internal UserInputResponse(string agentName, IList inputResponses) - { - this.AgentName = agentName; - this.InputResponses = inputResponses; - } - - /// - /// Factory method to create an from a - /// Ensures that all requests have a corresponding result. - /// - /// The input request. - /// One or more responses - /// An that can be provided to the workflow. - /// Not all have a corresponding . - public static UserInputResponse Create(UserInputRequest inputRequest, params IEnumerable inputResponses) - { - HashSet callIds = [.. inputRequest.InputRequests.OfType().Select(call => call.Id)]; - HashSet resultIds = [.. inputResponses.Select(call => call.Id)]; - - if (!callIds.SetEquals(resultIds)) - { - throw new DeclarativeActionException($"Missing responses for: {string.Join(",", callIds.Except(resultIds))}"); - } - - return new UserInputResponse(inputRequest.AgentName, [.. inputResponses]); - } -} diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Extensions/AgentProviderExtensions.cs b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Extensions/AgentProviderExtensions.cs index 5b6bbbc297..037665e8b8 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Extensions/AgentProviderExtensions.cs +++ b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Extensions/AgentProviderExtensions.cs @@ -1,24 +1,14 @@ // Copyright (c) Microsoft. All rights reserved. using System.Collections.Generic; -using System.Linq; using System.Threading; using System.Threading.Tasks; -using Azure.AI.Agents.Persistent; using Microsoft.Extensions.AI; namespace Microsoft.Agents.AI.Workflows.Declarative.Extensions; internal static class AgentProviderExtensions { - private static readonly HashSet s_failureStatus = - [ - Azure.AI.Agents.Persistent.RunStatus.Failed, - Azure.AI.Agents.Persistent.RunStatus.Cancelled, - Azure.AI.Agents.Persistent.RunStatus.Cancelling, - Azure.AI.Agents.Persistent.RunStatus.Expired, - ]; - public static async ValueTask InvokeAgentAsync( this WorkflowAgentProvider agentProvider, string executorId, @@ -26,27 +16,11 @@ public static async ValueTask InvokeAgentAsync( string agentName, string? conversationId, bool autoSend, - string? additionalInstructions = null, IEnumerable? inputMessages = null, + IDictionary? inputArguments = null, CancellationToken cancellationToken = default) { - // Get the specified agent. - AIAgent agent = await agentProvider.GetAgentAsync(agentName, cancellationToken).ConfigureAwait(false); - - // Prepare the run options. - ChatClientAgentRunOptions options = - new( - new ChatOptions() - { - ConversationId = conversationId, - Instructions = additionalInstructions, - }); - - // Initialize the agent thread. - IAsyncEnumerable agentUpdates = - inputMessages is not null ? - agent.RunStreamingAsync([.. inputMessages], null, options, cancellationToken) : - agent.RunStreamingAsync(null, options, cancellationToken); + IAsyncEnumerable agentUpdates = agentProvider.InvokeAgentAsync(agentName, null, conversationId, inputMessages, inputArguments, cancellationToken); // Enable "autoSend" behavior if this is the workflow conversation. bool isWorkflowConversation = context.IsWorkflowConversation(conversationId, out string? workflowConversationId); @@ -60,13 +34,6 @@ inputMessages is not null ? updates.Add(update); - if (update.RawRepresentation is ChatResponseUpdate chatUpdate && - chatUpdate.RawRepresentation is RunUpdate runUpdate && - s_failureStatus.Contains(runUpdate.Value.Status)) - { - throw new DeclarativeActionException($"Unexpected failure invoking agent, run {runUpdate.Value.Status}: {agent.Name ?? agent.Id} [{runUpdate.Value.Id}/{conversationId}]"); - } - if (autoSend) { await context.AddEventAsync(new AgentRunUpdateEvent(executorId, update), cancellationToken).ConfigureAwait(false); @@ -80,16 +47,10 @@ chatUpdate.RawRepresentation is RunUpdate runUpdate && await context.AddEventAsync(new AgentRunResponseEvent(executorId, response), cancellationToken).ConfigureAwait(false); } + // If autoSend is enabled and this is not the workflow conversation, copy messages to the workflow conversation. if (autoSend && !isWorkflowConversation && workflowConversationId is not null) { - // Copy messages with content that aren't function calls or results. - IEnumerable messages = - response.Messages.Where( - message => - !string.IsNullOrEmpty(message.Text) && - !message.Contents.OfType().Any() && - !message.Contents.OfType().Any()); - foreach (ChatMessage message in messages) + foreach (ChatMessage message in response.Messages) { await agentProvider.CreateMessageAsync(workflowConversationId, message, cancellationToken).ConfigureAwait(false); } diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Extensions/ChatMessageExtensions.cs b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Extensions/ChatMessageExtensions.cs index 279fec3e6d..1aa9a6ef71 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Extensions/ChatMessageExtensions.cs +++ b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Extensions/ChatMessageExtensions.cs @@ -20,7 +20,7 @@ public static TableValue ToTable(this IEnumerable messages) => public static IEnumerable? ToChatMessages(this DataValue? messages) { - if (messages is null || messages is BlankDataValue) + if (messages is null or BlankDataValue) { return null; } diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Extensions/DataValueExtensions.cs b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Extensions/DataValueExtensions.cs index a520593144..9d4d18db73 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Extensions/DataValueExtensions.cs +++ b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Extensions/DataValueExtensions.cs @@ -117,7 +117,7 @@ public static Type ToClrType(this DataType type) => public static IList? AsList(this DataValue? value) { - if (value is null || value is BlankDataValue) + if (value is null or BlankDataValue) { return null; } diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Extensions/FormulaValueExtensions.cs b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Extensions/FormulaValueExtensions.cs index 108dca7682..3e397f3e87 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Extensions/FormulaValueExtensions.cs +++ b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Extensions/FormulaValueExtensions.cs @@ -163,6 +163,24 @@ IEnumerable GetFields() } } } + + public static JsonNode ToJson(this FormulaValue value) => + value switch + { + BooleanValue booleanValue => JsonValue.Create(booleanValue.Value), + DecimalValue decimalValue => JsonValue.Create(decimalValue.Value), + NumberValue numberValue => JsonValue.Create(numberValue.Value), + DateValue dateValue => JsonValue.Create(dateValue.GetConvertedValue(TimeZoneInfo.Utc)), + DateTimeValue datetimeValue => JsonValue.Create(datetimeValue.GetConvertedValue(TimeZoneInfo.Utc)), + TimeValue timeValue => JsonValue.Create($"{timeValue.Value}"), + StringValue stringValue => JsonValue.Create(stringValue.Value), + GuidValue guidValue => JsonValue.Create(guidValue.Value), + RecordValue recordValue => recordValue.ToJson(), + TableValue tableValue => tableValue.ToJson(), + BlankValue => JsonValue.Create(string.Empty), + _ => $"[{value.GetType().Name}]", + }; + public static RecordValue ToRecord(this Dictionary value) => FormulaValue.NewRecordFromFields( value.Select( @@ -256,23 +274,6 @@ private static TableValue ToTableOfRecords(this IEnumerable list) private static KeyValuePair GetKeyValuePair(this NamedValue value) => new(value.Name, value.Value.ToDataValue()); - private static JsonNode ToJson(this FormulaValue value) => - value switch - { - BooleanValue booleanValue => JsonValue.Create(booleanValue.Value), - DecimalValue decimalValue => JsonValue.Create(decimalValue.Value), - NumberValue numberValue => JsonValue.Create(numberValue.Value), - DateValue dateValue => JsonValue.Create(dateValue.GetConvertedValue(TimeZoneInfo.Utc)), - DateTimeValue datetimeValue => JsonValue.Create(datetimeValue.GetConvertedValue(TimeZoneInfo.Utc)), - TimeValue timeValue => JsonValue.Create($"{timeValue.Value}"), - StringValue stringValue => JsonValue.Create(stringValue.Value), - GuidValue guidValue => JsonValue.Create(guidValue.Value), - RecordValue recordValue => recordValue.ToJson(), - TableValue tableValue => tableValue.ToJson(), - BlankValue => JsonValue.Create(string.Empty), - _ => $"[{value.GetType().Name}]", - }; - private static JsonArray ToJson(this TableValue value) { return new([.. GetJsonElements()]); diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Extensions/JsonDocumentExtensions.cs b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Extensions/JsonDocumentExtensions.cs index d3a4ef9cbc..9d3c5e335d 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Extensions/JsonDocumentExtensions.cs +++ b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Extensions/JsonDocumentExtensions.cs @@ -1,6 +1,7 @@ // Copyright (c) Microsoft. All rights reserved. using System; +using System.Collections.Frozen; using System.Collections.Generic; using System.Globalization; using System.Linq; @@ -43,16 +44,28 @@ internal static class JsonDocumentExtensions private static Dictionary ParseRecord(this JsonElement currentElement, VariableType targetType) { - if (targetType.Schema is null) - { - throw new DeclarativeActionException($"Object schema not defined for. {targetType.Type.Name}."); - } + IEnumerable> keyValuePairs = + targetType.Schema is null ? + ParseValues() : + ParseSchema(targetType.Schema); - return ParseValues().ToDictionary(kvp => kvp.Key, kvp => kvp.Value); + return keyValuePairs.ToDictionary(kvp => kvp.Key, kvp => kvp.Value); IEnumerable> ParseValues() { - foreach (KeyValuePair property in targetType.Schema) + foreach (JsonProperty objectProperty in currentElement.EnumerateObject()) + { + if (!objectProperty.Value.TryParseValue(targetType: null, out object? parsedValue)) + { + throw new DeclarativeActionException($"Unsupported data type '{objectProperty.Value.ValueKind}' for property '{objectProperty.Name}'"); + } + yield return new KeyValuePair(objectProperty.Name, parsedValue); + } + } + + IEnumerable> ParseSchema(FrozenDictionary schema) + { + foreach (KeyValuePair property in schema) { object? parsedValue = null; if (!currentElement.TryGetProperty(property.Key, out JsonElement propertyElement)) @@ -131,22 +144,22 @@ VariableType DetermineElementType() return value; } - private static bool TryParseValue(this JsonElement propertyElement, VariableType targetType, out object? value) => + private static bool TryParseValue(this JsonElement propertyElement, VariableType? targetType, out object? value) => propertyElement.ValueKind switch { - JsonValueKind.String => TryParseString(propertyElement, targetType.Type, out value), - JsonValueKind.Number => TryParseNumber(propertyElement, targetType.Type, out value), + JsonValueKind.String => TryParseString(propertyElement, targetType?.Type, out value), + JsonValueKind.Number => TryParseNumber(propertyElement, targetType?.Type, out value), JsonValueKind.True or JsonValueKind.False => TryParseBoolean(propertyElement, out value), JsonValueKind.Object => TryParseObject(propertyElement, targetType, out value), JsonValueKind.Array => TryParseList(propertyElement, targetType, out value), - JsonValueKind.Null => TryParseNull(targetType.Type, out value), + JsonValueKind.Null => TryParseNull(targetType?.Type, out value), _ => throw new DeclarativeActionException($"JSON element of type {propertyElement.ValueKind} is not supported."), }; - private static bool TryParseNull(Type valueType, out object? value) + private static bool TryParseNull(Type? valueType, out object? value) { // If the target type is not nullable, we cannot assign null to it - if (!valueType.IsNullable()) + if (valueType?.IsNullable() == false) { value = null; return false; @@ -170,7 +183,7 @@ private static bool TryParseBoolean(JsonElement propertyElement, out object? val } } - private static bool TryParseString(JsonElement propertyElement, Type valueType, out object? value) + private static bool TryParseString(JsonElement propertyElement, Type? valueType, out object? value) { try { @@ -178,23 +191,30 @@ private static bool TryParseString(JsonElement propertyElement, Type valueType, if (propertyValue is null) { value = null; - return valueType.IsNullable(); // Parse fails if value is null and requested type is not. + return valueType?.IsNullable() ?? false; // Parse fails if value is null and requested type is not. } - switch (valueType) + if (valueType is null) + { + value = propertyValue; + } + else { - case Type targetType when targetType == typeof(string): - value = propertyValue; - break; - case Type targetType when targetType == typeof(DateTime): - value = DateTime.Parse(propertyValue, provider: null, styles: DateTimeStyles.RoundtripKind); - break; - case Type targetType when targetType == typeof(TimeSpan): - value = TimeSpan.Parse(propertyValue); - break; - default: - value = null; - return false; + switch (valueType) + { + case Type targetType when targetType == typeof(string): + value = propertyValue; + break; + case Type targetType when targetType == typeof(DateTime): + value = DateTime.Parse(propertyValue, provider: null, styles: DateTimeStyles.RoundtripKind); + break; + case Type targetType when targetType == typeof(TimeSpan): + value = TimeSpan.Parse(propertyValue); + break; + default: + value = null; + return false; + } } return true; @@ -206,7 +226,7 @@ private static bool TryParseString(JsonElement propertyElement, Type valueType, } } - private static bool TryParseNumber(JsonElement element, Type valueType, out object? value) + private static bool TryParseNumber(JsonElement element, Type? valueType, out object? value) { // Try parsing as integer types first (most precise representation) if (element.TryGetInt32(out int intValue)) @@ -234,8 +254,14 @@ private static bool TryParseNumber(JsonElement element, Type valueType, out obje value = null; return false; - static bool ConvertToExpectedType(Type valueType, object sourceValue, out object? value) + static bool ConvertToExpectedType(Type? valueType, object sourceValue, out object? value) { + if (valueType is null) + { + value = sourceValue; + return true; + } + try { value = Convert.ChangeType(sourceValue, valueType); @@ -249,23 +275,17 @@ static bool ConvertToExpectedType(Type valueType, object sourceValue, out object } } - private static bool TryParseObject(JsonElement propertyElement, VariableType targetType, out object? value) + private static bool TryParseObject(JsonElement propertyElement, VariableType? targetType, out object? value) { - if (!targetType.HasSchema) - { - value = null; - return false; - } - - value = propertyElement.ParseRecord(targetType); + value = propertyElement.ParseRecord(targetType ?? VariableType.RecordType); return true; } - private static bool TryParseList(JsonElement propertyElement, VariableType targetType, out object? value) + private static bool TryParseList(JsonElement propertyElement, VariableType? targetType, out object? value) { try { - value = ParseTable(propertyElement, targetType); + value = ParseTable(propertyElement, targetType ?? VariableType.ListType); return true; } catch diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Extensions/PortableValueExtensions.cs b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Extensions/PortableValueExtensions.cs index 17e7579d9f..7ef09d2b85 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Extensions/PortableValueExtensions.cs +++ b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Extensions/PortableValueExtensions.cs @@ -40,6 +40,12 @@ _ when value.IsType(out TimeSpan timeValue) => FormulaValue.New(timeValue), private static TableValue ToTable(this PortableValue[] values) { FormulaValue[] formulaValues = values.Select(value => value.ToFormula()).ToArray(); + + if (formulaValues.Length == 0) + { + return FormulaValue.NewTable(RecordType.Empty()); + } + if (formulaValues[0] is RecordValue recordValue) { return FormulaValue.NewTable(ParseRecordType(recordValue), formulaValues.OfType()); diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Interpreter/DeclarativeActionExecutor.cs b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Interpreter/DeclarativeActionExecutor.cs index 704a555159..2ad605803e 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Interpreter/DeclarativeActionExecutor.cs +++ b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Interpreter/DeclarativeActionExecutor.cs @@ -24,7 +24,6 @@ internal abstract class DeclarativeActionExecutor(TAction model, Workfl internal abstract class DeclarativeActionExecutor : Executor, IResettableExecutor, IModeledAction { - private string? _parentId; private readonly WorkflowFormulaState _state; protected DeclarativeActionExecutor(DialogAction model, WorkflowFormulaState state) @@ -42,7 +41,7 @@ protected DeclarativeActionExecutor(DialogAction model, WorkflowFormulaState sta public DialogAction Model { get; } - public string ParentId => this._parentId ??= this.Model.GetParentId() ?? WorkflowActionVisitor.Steps.Root(); + public string ParentId { get => field ??= this.Model.GetParentId() ?? WorkflowActionVisitor.Steps.Root(); } public RecalcEngine Engine => this._state.Engine; diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Interpreter/DeclarativeWorkflowContext.cs b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Interpreter/DeclarativeWorkflowContext.cs index 60e50e6abe..9a8d9da707 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Interpreter/DeclarativeWorkflowContext.cs +++ b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Interpreter/DeclarativeWorkflowContext.cs @@ -77,7 +77,7 @@ public async ValueTask QueueStateUpdateAsync(string key, T? value, string? sc this.State.Bind(); } - private bool IsManagedScope(string? scopeName) => scopeName is not null && VariableScopeNames.IsValidName(scopeName); + private static bool IsManagedScope(string? scopeName) => scopeName is not null && VariableScopeNames.IsValidName(scopeName); /// public async ValueTask ReadStateAsync(string key, string? scopeName = null, CancellationToken cancellationToken = default) @@ -86,7 +86,7 @@ public async ValueTask QueueStateUpdateAsync(string key, T? value, string? sc { // Not a managed scope, just pass through. This is valid when a declarative // workflow has been ejected to code (where DeclarativeWorkflowContext is also utilized). - _ when !this.IsManagedScope(scopeName) => await this.Source.ReadStateAsync(key, scopeName, cancellationToken).ConfigureAwait(false), + _ when !IsManagedScope(scopeName) => await this.Source.ReadStateAsync(key, scopeName, cancellationToken).ConfigureAwait(false), // Retrieve formula values directly from the managed state to avoid conversion. _ when typeof(TValue) == typeof(FormulaValue) => (TValue?)(object?)this.State.Get(key, scopeName), // Retrieve native types from the source context to avoid conversion. @@ -100,7 +100,7 @@ public async ValueTask ReadOrInitStateAsync(string key, Func await this.Source.ReadOrInitStateAsync(key, initialStateFactory, scopeName, cancellationToken).ConfigureAwait(false), + _ when !IsManagedScope(scopeName) => await this.Source.ReadOrInitStateAsync(key, initialStateFactory, scopeName, cancellationToken).ConfigureAwait(false), // Retrieve formula values directly from the managed state to avoid conversion. _ when typeof(TValue) == typeof(FormulaValue) => await EnsureFormulaValueAsync().ConfigureAwait(false), // Retrieve native types from the source context to avoid conversion. diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Interpreter/WorkflowActionVisitor.cs b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Interpreter/WorkflowActionVisitor.cs index de90787cfd..b6bcd458d8 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Interpreter/WorkflowActionVisitor.cs +++ b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Interpreter/WorkflowActionVisitor.cs @@ -137,18 +137,23 @@ protected override void Visit(ConditionGroup item) conditionItem.Accept(this); } - if (item.ElseActions?.Actions.Length > 0) + if (lastConditionItemId is not null) { - if (lastConditionItemId is not null) - { - // Create clean start for else action from prior conditions - this.RestartAfter(lastConditionItemId, action.Id); - } + // Create clean start for else action from prior conditions + this.RestartAfter(lastConditionItemId, action.Id); + } + if (item.ElseActions?.Actions.Length > 0) + { // Create conditional link for else action string stepId = ConditionGroupExecutor.Steps.Else(item); this._workflowModel.AddLink(action.Id, stepId, action.IsElse); } + else + { + string stepId = Steps.Post(action.Id); + this._workflowModel.AddLink(action.Id, stepId, action.IsElse); + } } protected override void Visit(GotoAction item) @@ -239,6 +244,7 @@ protected override void Visit(Question item) // Entry point for question QuestionExecutor action = new(item, this._workflowOptions.AgentProvider, this._workflowState); this.ContinueWith(action); + // Transition to post action if complete string postId = Steps.Post(action.Id); this._workflowModel.AddLink(action.Id, postId, QuestionExecutor.IsComplete); @@ -249,13 +255,13 @@ protected override void Visit(Question item) // Define input action string inputId = QuestionExecutor.Steps.Input(action.Id); - RequestPortAction inputPort = new(RequestPort.Create(inputId)); + RequestPortAction inputPort = new(RequestPort.Create(inputId)); this._workflowModel.AddNode(inputPort, action.ParentId); this._workflowModel.AddLinkFromPeer(action.ParentId, inputId); // Capture input response string captureId = QuestionExecutor.Steps.Capture(action.Id); - this.ContinueWith(new DelegateActionExecutor(captureId, this._workflowState, action.CaptureResponseAsync, emitResult: false), action.ParentId); + this.ContinueWith(new DelegateActionExecutor(captureId, this._workflowState, action.CaptureResponseAsync, emitResult: false), action.ParentId); // Transition to post action if complete this.ContinueWith(new DelegateActionExecutor(postId, this._workflowState, action.CompleteAsync), action.ParentId, QuestionExecutor.IsComplete); @@ -263,6 +269,24 @@ protected override void Visit(Question item) this._workflowModel.AddLink(captureId, prepareId, message => !QuestionExecutor.IsComplete(message)); } + protected override void Visit(RequestExternalInput item) + { + this.Trace(item); + + RequestExternalInputExecutor action = new(item, this._workflowOptions.AgentProvider, this._workflowState); + this.ContinueWith(action); + + // Define input action + string inputId = RequestExternalInputExecutor.Steps.Input(action.Id); + RequestPortAction inputPort = new(RequestPort.Create(inputId)); + this._workflowModel.AddNode(inputPort, action.ParentId); + this._workflowModel.AddLinkFromPeer(action.ParentId, inputId); + + // Capture input response + string captureId = RequestExternalInputExecutor.Steps.Capture(action.Id); + this.ContinueWith(new DelegateActionExecutor(captureId, this._workflowState, action.CaptureResponseAsync), action.ParentId); + } + protected override void Visit(EndDialog item) { this.Trace(item); @@ -285,6 +309,28 @@ protected override void Visit(EndConversation item) this.RestartAfter(action.Id, action.ParentId); } + protected override void Visit(CancelAllDialogs item) + { + this.Trace(item); + + // Represent action with default executor + DefaultActionExecutor action = new(item, this._workflowState); + this.ContinueWith(action); + // Define a clean-start to ensure "end" is not a source for any edge + this.RestartAfter(item.Id.Value, action.ParentId); + } + + protected override void Visit(CancelDialog item) + { + this.Trace(item); + + // Represent action with default executor + DefaultActionExecutor action = new(item, this._workflowState); + this.ContinueWith(action); + // Define a clean-start to ensure "end" is not a source for any edge + this.RestartAfter(action.Id, action.ParentId); + } + protected override void Visit(CreateConversation item) { this.Trace(item); @@ -318,33 +364,29 @@ protected override void Visit(InvokeAzureAgent item) this._workflowModel.AddLink(action.Id, postId, InvokeAzureAgentExecutor.RequiresNothing); // Define request-port for function calling action - string functionCallingPortId = InvokeAzureAgentExecutor.Steps.FunctionTool(action.Id); - RequestPortAction functionCallingPort = new(RequestPort.Create(functionCallingPortId)); - this._workflowModel.AddNode(functionCallingPort, action.ParentId); - this._workflowModel.AddLink(action.Id, functionCallingPort.Id, InvokeAzureAgentExecutor.RequiresFunctionCall); - - // Define request-port for user input, such as: mcp tool & function tool approval - string userInputPortId = InvokeAzureAgentExecutor.Steps.UserInput(action.Id); - RequestPortAction userInputPort = new(RequestPort.Create(userInputPortId)); - this._workflowModel.AddNode(userInputPort, action.ParentId); - this._workflowModel.AddLink(action.Id, userInputPortId, InvokeAzureAgentExecutor.RequiresUserInput); + string externalInputPortId = InvokeAzureAgentExecutor.Steps.ExternalInput(action.Id); + RequestPortAction externalInputPort = new(RequestPort.Create(externalInputPortId)); + this._workflowModel.AddNode(externalInputPort, action.ParentId); + this._workflowModel.AddLink(action.Id, externalInputPortId, InvokeAzureAgentExecutor.RequiresInput); // Request ports always transitions to resume string resumeId = InvokeAzureAgentExecutor.Steps.Resume(action.Id); - this._workflowModel.AddNode(new DelegateActionExecutor(resumeId, this._workflowState, action.ResumeAsync), action.ParentId); - this._workflowModel.AddLink(functionCallingPortId, resumeId); - this._workflowModel.AddLink(userInputPortId, resumeId); - // Transition to appropriate request port if more function calling is requested - this._workflowModel.AddLink(resumeId, functionCallingPortId, InvokeAzureAgentExecutor.RequiresFunctionCall); - // Transition to appropriate request port if more user input is requested - this._workflowModel.AddLink(resumeId, userInputPortId, InvokeAzureAgentExecutor.RequiresUserInput); + this._workflowModel.AddNode(new DelegateActionExecutor(resumeId, this._workflowState, action.ResumeAsync, emitResult: false), action.ParentId); + this._workflowModel.AddLink(externalInputPortId, resumeId); // Transition to post action if complete this._workflowModel.AddLink(resumeId, postId, InvokeAzureAgentExecutor.RequiresNothing); + // Transition to request port if more input is required + this._workflowModel.AddLink(resumeId, externalInputPortId, InvokeAzureAgentExecutor.RequiresInput); // Define post action this._workflowModel.AddNode(new DelegateActionExecutor(postId, this._workflowState, action.CompleteAsync), action.ParentId); } + protected override void Visit(InvokeAzureResponse item) + { + this.NotSupported(item); + } + protected override void Visit(RetrieveConversationMessage item) { this.Trace(item); @@ -462,10 +504,6 @@ protected override void Visit(SendActivity item) protected override void Visit(ReplaceDialog item) => this.NotSupported(item); - protected override void Visit(CancelAllDialogs item) => this.NotSupported(item); - - protected override void Visit(CancelDialog item) => this.NotSupported(item); - protected override void Visit(EmitEvent item) => this.NotSupported(item); protected override void Visit(GetConversationMembers item) => this.NotSupported(item); diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Interpreter/WorkflowTemplateVisitor.cs b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Interpreter/WorkflowTemplateVisitor.cs index 4822a70024..5d6a7b3384 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Interpreter/WorkflowTemplateVisitor.cs +++ b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Interpreter/WorkflowTemplateVisitor.cs @@ -210,9 +210,11 @@ protected override void Visit(ContinueLoop item) protected override void Visit(Question item) { this.NotSupported(item); - //this.Trace(item); + } - //this.ContinueWith(new QuestionTemplate(item)); + protected override void Visit(RequestExternalInput item) + { + this.NotSupported(item); } protected override void Visit(EndDialog item) @@ -237,6 +239,24 @@ protected override void Visit(EndConversation item) this.RestartAfter(action.Id, action.ParentId); } + protected override void Visit(CancelAllDialogs item) + { + // Represent action with default executor + DefaultTemplate action = new(item, this._rootId); + this.ContinueWith(action); + // Define a clean-start to ensure "end" is not a source for any edge + this.RestartAfter(action.Id, action.ParentId); + } + + protected override void Visit(CancelDialog item) + { + // Represent action with default executor + DefaultTemplate action = new(item, this._rootId); + this.ContinueWith(action); + // Define a clean-start to ensure "end" is not a source for any edge + this.RestartAfter(action.Id, action.ParentId); + } + protected override void Visit(CreateConversation item) { this.Trace(item); @@ -265,6 +285,11 @@ protected override void Visit(InvokeAzureAgent item) this.ContinueWith(new InvokeAzureAgentTemplate(item)); } + protected override void Visit(InvokeAzureResponse item) + { + this.NotSupported(item); + } + protected override void Visit(RetrieveConversationMessage item) { this.Trace(item); @@ -317,17 +342,11 @@ protected override void Visit(ResetVariable item) protected override void Visit(EditTable item) { this.NotSupported(item); - //this.Trace(item); - - //this.ContinueWith(new EditTableTemplate(item)); } protected override void Visit(EditTableV2 item) { this.NotSupported(item); - //this.Trace(item); - - //this.ContinueWith(new EditTableV2Template(item)); } protected override void Visit(ParseValue item) @@ -384,10 +403,6 @@ protected override void Visit(SendActivity item) protected override void Visit(ReplaceDialog item) => this.NotSupported(item); - protected override void Visit(CancelAllDialogs item) => this.NotSupported(item); - - protected override void Visit(CancelDialog item) => this.NotSupported(item); - protected override void Visit(EmitEvent item) => this.NotSupported(item); protected override void Visit(GetConversationMembers item) => this.NotSupported(item); diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Kit/AgentExecutor.cs b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Kit/AgentExecutor.cs index 8189e9b8aa..45a5b47bd8 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Kit/AgentExecutor.cs +++ b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Kit/AgentExecutor.cs @@ -23,7 +23,6 @@ public abstract class AgentExecutor(string id, FormulaSession session, WorkflowA /// The name or identifier of the agent. /// The identifier of the conversation. /// Send the agent's response as workflow output. (default: true). - /// Optional additional instructions to the agent. /// Optional messages to add to the conversation prior to invocation. /// A token that can be used to observe cancellation. /// @@ -32,8 +31,7 @@ protected ValueTask InvokeAgentAsync( string agentName, string? conversationId, bool autoSend, - string? additionalInstructions = null, IEnumerable? inputMessages = null, CancellationToken cancellationToken = default) - => agentProvider.InvokeAgentAsync(this.Id, context, agentName, conversationId, autoSend, additionalInstructions, inputMessages, cancellationToken); + => agentProvider.InvokeAgentAsync(this.Id, context, agentName, conversationId, autoSend, inputMessages, inputArguments: null, cancellationToken); } diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Microsoft.Agents.AI.Workflows.Declarative.csproj b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Microsoft.Agents.AI.Workflows.Declarative.csproj index 7db0b0d941..0b3f41ec9b 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Microsoft.Agents.AI.Workflows.Declarative.csproj +++ b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/Microsoft.Agents.AI.Workflows.Declarative.csproj @@ -1,10 +1,8 @@  - $(ProjectsTargetFrameworks) - $(ProjectsDebugTargetFrameworks) preview - $(NoWarn);MEAI001 + $(NoWarn);MEAI001;OPENAI001 @@ -22,7 +20,6 @@ - @@ -35,7 +32,6 @@ - diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/ObjectModel/AddConversationMessageExecutor.cs b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/ObjectModel/AddConversationMessageExecutor.cs index ef64625f6e..632f462758 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/ObjectModel/AddConversationMessageExecutor.cs +++ b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/ObjectModel/AddConversationMessageExecutor.cs @@ -19,6 +19,7 @@ internal sealed class AddConversationMessageExecutor(AddConversationMessage mode { Throw.IfNull(this.Model.ConversationId, $"{nameof(this.Model)}.{nameof(this.Model.ConversationId)}"); string conversationId = this.Evaluator.GetValue(this.Model.ConversationId).Value; + bool isWorkflowConversation = context.IsWorkflowConversation(conversationId, out string? _); ChatMessage newMessage = new(this.Model.Role.Value.ToChatRole(), [.. this.GetContent()]) { AdditionalProperties = this.GetMetadata() }; @@ -27,6 +28,11 @@ internal sealed class AddConversationMessageExecutor(AddConversationMessage mode await this.AssignAsync(this.Model.Message?.Path, newMessage.ToRecord(), context).ConfigureAwait(false); + if (isWorkflowConversation) + { + await context.AddEventAsync(new AgentRunResponseEvent(this.Id, new AgentRunResponse(newMessage)), cancellationToken).ConfigureAwait(false); + } + return default; } diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/ObjectModel/CopyConversationMessagesExecutor.cs b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/ObjectModel/CopyConversationMessagesExecutor.cs index e54e294730..01b1bab496 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/ObjectModel/CopyConversationMessagesExecutor.cs +++ b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/ObjectModel/CopyConversationMessagesExecutor.cs @@ -20,6 +20,7 @@ internal sealed class CopyConversationMessagesExecutor(CopyConversationMessages { Throw.IfNull(this.Model.ConversationId, $"{nameof(this.Model)}.{nameof(this.Model.ConversationId)}"); string conversationId = this.Evaluator.GetValue(this.Model.ConversationId).Value; + bool isWorkflowConversation = context.IsWorkflowConversation(conversationId, out string? _); IEnumerable? inputMessages = this.GetInputMessages(); @@ -29,6 +30,11 @@ internal sealed class CopyConversationMessagesExecutor(CopyConversationMessages { await agentProvider.CreateMessageAsync(conversationId, message, cancellationToken).ConfigureAwait(false); } + + if (isWorkflowConversation) + { + await context.AddEventAsync(new AgentRunResponseEvent(this.Id, new AgentRunResponse([.. inputMessages])), cancellationToken).ConfigureAwait(false); + } } return default; diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/ObjectModel/InvokeAzureAgentExecutor.cs b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/ObjectModel/InvokeAzureAgentExecutor.cs index 8c4c9eee61..ed069a9b78 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/ObjectModel/InvokeAzureAgentExecutor.cs +++ b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/ObjectModel/InvokeAzureAgentExecutor.cs @@ -2,6 +2,7 @@ using System.Collections.Generic; using System.Linq; +using System.Text.Json; using System.Threading; using System.Threading.Tasks; using Microsoft.Agents.AI.Workflows.Declarative.Events; @@ -21,14 +22,11 @@ internal sealed class InvokeAzureAgentExecutor(InvokeAzureAgent model, WorkflowA { public static class Steps { - public static string UserInput(string id) => $"{id}_{nameof(UserInput)}"; - public static string FunctionTool(string id) => $"{id}_{nameof(FunctionTool)}"; + public static string ExternalInput(string id) => $"{id}_{nameof(ExternalInput)}"; public static string Resume(string id) => $"{id}_{nameof(Resume)}"; } - public static bool RequiresFunctionCall(object? message) => message is AgentFunctionToolRequest; - - public static bool RequiresUserInput(object? message) => message is UserInputRequest; + public static bool RequiresInput(object? message) => message is ExternalInputRequest; public static bool RequiresNothing(object? message) => message is ActionExecutorResult; @@ -46,8 +44,11 @@ public static class Steps return default; } - public ValueTask ResumeAsync(IWorkflowContext context, AgentFunctionToolResponse message, CancellationToken cancellationToken) => - this.InvokeAgentAsync(context, [message.FunctionResults.ToChatMessage()], cancellationToken); + public async ValueTask ResumeAsync(IWorkflowContext context, ExternalInputResponse response, CancellationToken cancellationToken) + { + await context.SetLastMessageAsync(response.Messages.Last()).ConfigureAwait(false); + await this.InvokeAgentAsync(context, response.Messages, cancellationToken).ConfigureAwait(false); + } public async ValueTask CompleteAsync(IWorkflowContext context, ActionExecutorResult message, CancellationToken cancellationToken) { @@ -58,40 +59,69 @@ private async ValueTask InvokeAgentAsync(IWorkflowContext context, IEnumerable? inputParameters = this.GetStructuredInputs(); + AgentRunResponse agentResponse = await agentProvider.InvokeAgentAsync(this.Id, context, agentName, conversationId, autoSend, messages, inputParameters, cancellationToken).ConfigureAwait(false); - bool isComplete = true; + ChatMessage[] actionableMessages = FilterActionableContent(agentResponse).ToArray(); + if (actionableMessages.Length > 0) + { + AgentRunResponse filteredResponse = + new(actionableMessages) + { + AdditionalProperties = agentResponse.AdditionalProperties, + AgentId = agentResponse.AgentId, + CreatedAt = agentResponse.CreatedAt, + ResponseId = agentResponse.ResponseId, + Usage = agentResponse.Usage, + }; + await context.SendMessageAsync(new ExternalInputRequest(filteredResponse), cancellationToken).ConfigureAwait(false); + return; + } - AgentRunResponse agentResponse = await agentProvider.InvokeAgentAsync(this.Id, context, agentName, conversationId, autoSend, additionalInstructions, messages, cancellationToken).ConfigureAwait(false); + await this.AssignAsync(this.AgentOutput?.Messages?.Path, agentResponse.Messages.ToTable(), context).ConfigureAwait(false); - if (string.IsNullOrEmpty(agentResponse.Text)) + // Attempt to parse the last message as JSON and assign to the response object variable. + try { - // Identify function calls that have no associated result. - List inputRequests = GetUserInputRequests(agentResponse); - if (inputRequests.Count > 0) - { - isComplete = false; - UserInputRequest approvalRequest = new(agentName, inputRequests.OfType().ToArray()); - await context.SendMessageAsync(approvalRequest, cancellationToken).ConfigureAwait(false); - } + JsonDocument jsonDocument = JsonDocument.Parse(agentResponse.Messages.Last().Text); + Dictionary objectProperties = jsonDocument.ParseRecord(VariableType.RecordType); + await this.AssignAsync(this.AgentOutput?.ResponseObject?.Path, objectProperties.ToFormula(), context).ConfigureAwait(false); + } + catch + { + // Not valid json, skip assignment. + } - // Identify function calls that have no associated result. - List functionCalls = GetOrphanedFunctionCalls(agentResponse); - if (functionCalls.Count > 0) + if (this.Model.Input?.ExternalLoop?.When is not null) + { + bool requestInput = this.Evaluator.GetValue(this.Model.Input.ExternalLoop.When).Value; + if (requestInput) { - isComplete = false; - AgentFunctionToolRequest toolRequest = new(agentName, functionCalls); - await context.SendMessageAsync(toolRequest, cancellationToken).ConfigureAwait(false); + ExternalInputRequest inputRequest = new(agentResponse); + await context.SendMessageAsync(inputRequest, cancellationToken).ConfigureAwait(false); + return; } } - if (isComplete) + await context.SendResultMessageAsync(this.Id, result: null, cancellationToken).ConfigureAwait(false); + } + + private Dictionary? GetStructuredInputs() + { + Dictionary? inputs = null; + + if (this.AgentInput?.Arguments is not null) { - await context.SendResultMessageAsync(this.Id, result: null, cancellationToken).ConfigureAwait(false); + inputs = []; + + foreach (KeyValuePair argument in this.AgentInput.Arguments) + { + inputs[argument.Key] = this.Evaluator.GetValue(argument.Value).Value.ToObject(); + } } - await this.AssignAsync(this.AgentOutput?.Messages?.Path, agentResponse.Messages.ToTable(), context).ConfigureAwait(false); + return inputs; } private IEnumerable? GetInputMessages() @@ -107,7 +137,7 @@ private async ValueTask InvokeAgentAsync(IWorkflowContext context, IEnumerable GetOrphanedFunctionCalls(AgentRunResponse agentResponse) + private static IEnumerable FilterActionableContent(AgentRunResponse agentResponse) { HashSet functionResultIds = [.. agentResponse.Messages @@ -117,21 +147,21 @@ [.. agentResponse.Messages .OfType() .Select(functionCall => functionCall.CallId))]; - List functionCalls = []; - foreach (FunctionCallContent functionCall in agentResponse.Messages.SelectMany(m => m.Contents.OfType())) + foreach (ChatMessage responseMessage in agentResponse.Messages) { - if (!functionResultIds.Contains(functionCall.CallId)) + if (responseMessage.Contents.Any(content => content is UserInputRequestContent)) { - functionCalls.Add(functionCall); + yield return responseMessage; + continue; } - } - return functionCalls; + if (responseMessage.Contents.OfType().Any(functionCall => !functionResultIds.Contains(functionCall.CallId))) + { + yield return responseMessage; + } + } } - private static List GetUserInputRequests(AgentRunResponse agentResponse) => - agentResponse.Messages.SelectMany(m => m.Contents.OfType()).ToList(); - private string? GetConversationId() { if (this.Model.ConversationId is null) @@ -149,18 +179,6 @@ private string GetAgentName() => this.AgentUsage.Name, $"{nameof(this.Model)}.{nameof(this.Model.Agent)}.{nameof(this.Model.Agent.Name)}")).Value; - private string? GetAdditionalInstructions() - { - string? additionalInstructions = null; - - if (this.AgentInput?.AdditionalInstructions is not null) - { - additionalInstructions = this.Engine.Format(this.AgentInput.AdditionalInstructions); - } - - return additionalInstructions; - } - private bool GetAutoSendValue() { if (this.AgentOutput?.AutoSend is null) diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/ObjectModel/QuestionExecutor.cs b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/ObjectModel/QuestionExecutor.cs index 145567f2ce..40dc5ce31e 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/ObjectModel/QuestionExecutor.cs +++ b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/ObjectModel/QuestionExecutor.cs @@ -1,5 +1,6 @@ // Copyright (c) Microsoft. All rights reserved. +using System.Linq; using System.Threading; using System.Threading.Tasks; using Microsoft.Agents.AI.Workflows.Declarative.Entities; @@ -75,22 +76,22 @@ public static bool IsComplete(object? message) public async ValueTask PrepareResponseAsync(IWorkflowContext context, ActionExecutorResult message, CancellationToken cancellationToken) { int count = await this._promptCount.ReadAsync(context).ConfigureAwait(false); - AnswerRequest inputRequest = new(this.FormatPrompt(this.Model.Prompt)); + ExternalInputRequest inputRequest = new(this.FormatPrompt(this.Model.Prompt)); await context.SendMessageAsync(inputRequest, cancellationToken).ConfigureAwait(false); await this._promptCount.WriteAsync(context, count + 1).ConfigureAwait(false); } - public async ValueTask CaptureResponseAsync(IWorkflowContext context, AnswerResponse message, CancellationToken cancellationToken) + public async ValueTask CaptureResponseAsync(IWorkflowContext context, ExternalInputResponse response, CancellationToken cancellationToken) { FormulaValue? extractedValue = null; - if (message.Value is null) + if (!response.HasMessages) { string unrecognizedResponse = this.FormatPrompt(this.Model.UnrecognizedPrompt); await context.AddEventAsync(new MessageActivityEvent(unrecognizedResponse.Trim()), cancellationToken).ConfigureAwait(false); } else { - EntityExtractionResult entityResult = EntityExtractor.Parse(this.Model.Entity, message.Value.Text); + EntityExtractionResult entityResult = EntityExtractor.Parse(this.Model.Entity, string.Concat(response.Messages.Select(message => message.Text))); if (entityResult.IsValid) { extractedValue = entityResult.Value; @@ -121,7 +122,7 @@ public async ValueTask CaptureResponseAsync(IWorkflowContext context, AnswerResp if (workflowConversationId is not null) { // Input message always defined if values has been extracted. - ChatMessage input = message.Value!; + ChatMessage input = response.Messages.Last(); await agentProvider.CreateMessageAsync(workflowConversationId, input, cancellationToken).ConfigureAwait(false); await context.SetLastMessageAsync(input).ConfigureAwait(false); } diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/ObjectModel/RequestExternalInputExecutor.cs b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/ObjectModel/RequestExternalInputExecutor.cs new file mode 100644 index 0000000000..2c35dc18e9 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/ObjectModel/RequestExternalInputExecutor.cs @@ -0,0 +1,49 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Linq; +using System.Threading; +using System.Threading.Tasks; +using Microsoft.Agents.AI.Workflows.Declarative.Events; +using Microsoft.Agents.AI.Workflows.Declarative.Extensions; +using Microsoft.Agents.AI.Workflows.Declarative.Interpreter; +using Microsoft.Agents.AI.Workflows.Declarative.PowerFx; +using Microsoft.Bot.ObjectModel; +using Microsoft.Extensions.AI; + +namespace Microsoft.Agents.AI.Workflows.Declarative.ObjectModel; + +internal sealed class RequestExternalInputExecutor(RequestExternalInput model, WorkflowAgentProvider agentProvider, WorkflowFormulaState state) + : DeclarativeActionExecutor(model, state) +{ + public static class Steps + { + public static string Input(string id) => $"{id}_{nameof(Input)}"; + public static string Capture(string id) => $"{id}_{nameof(Capture)}"; + } + + protected override bool IsDiscreteAction => false; + protected override bool EmitResultEvent => false; + + protected override async ValueTask ExecuteAsync(IWorkflowContext context, CancellationToken cancellationToken = default) + { + ExternalInputRequest inputRequest = new(new AgentRunResponse()); + + await context.SendMessageAsync(inputRequest, cancellationToken).ConfigureAwait(false); + + return default; + } + + public async ValueTask CaptureResponseAsync(IWorkflowContext context, ExternalInputResponse response, CancellationToken cancellationToken) + { + string? workflowConversationId = context.GetWorkflowConversation(); + if (workflowConversationId is not null) + { + foreach (ChatMessage inputMessage in response.Messages) + { + await agentProvider.CreateMessageAsync(workflowConversationId, inputMessage, cancellationToken).ConfigureAwait(false); + } + } + await context.SetLastMessageAsync(response.Messages.Last()).ConfigureAwait(false); + await this.AssignAsync(this.Model.Variable?.Path, response.Messages.ToFormula(), context).ConfigureAwait(false); + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/ObjectModel/SetVariableExecutor.cs b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/ObjectModel/SetVariableExecutor.cs index 449e982982..81ed6e30d3 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/ObjectModel/SetVariableExecutor.cs +++ b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/ObjectModel/SetVariableExecutor.cs @@ -8,7 +8,6 @@ using Microsoft.Bot.ObjectModel; using Microsoft.Bot.ObjectModel.Abstractions; using Microsoft.PowerFx.Types; -using Microsoft.Shared.Diagnostics; namespace Microsoft.Agents.AI.Workflows.Declarative.ObjectModel; @@ -17,17 +16,15 @@ internal sealed class SetVariableExecutor(SetVariable model, WorkflowFormulaStat { protected override async ValueTask ExecuteAsync(IWorkflowContext context, CancellationToken cancellationToken = default) { - PropertyPath variablePath = Throw.IfNull(this.Model.Variable?.Path, $"{nameof(this.Model)}.{nameof(model.Variable)}"); - if (this.Model.Value is null) { - await this.AssignAsync(variablePath, FormulaValue.NewBlank(), context).ConfigureAwait(false); + await this.AssignAsync(this.Model.Variable?.Path, FormulaValue.NewBlank(), context).ConfigureAwait(false); } else { EvaluationResult expressionResult = this.Evaluator.GetValue(this.Model.Value); - await this.AssignAsync(variablePath, expressionResult.Value.ToFormula(), context).ConfigureAwait(false); + await this.AssignAsync(this.Model.Variable?.Path, expressionResult.Value.ToFormula(), context).ConfigureAwait(false); } return default; diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/PowerFx/Functions/AgentMessage.cs b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/PowerFx/Functions/AgentMessage.cs new file mode 100644 index 0000000000..927a842478 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/PowerFx/Functions/AgentMessage.cs @@ -0,0 +1,15 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Microsoft.Extensions.AI; +using Microsoft.PowerFx.Types; + +namespace Microsoft.Agents.AI.Workflows.Declarative.PowerFx.Functions; + +internal sealed class AgentMessage : MessageFunction +{ + public const string FunctionName = nameof(AgentMessage); + + public AgentMessage() : base(FunctionName) { } + + public static FormulaValue Execute(StringValue input) => Create(ChatRole.Assistant, input); +} diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/PowerFx/Functions/MessageFunction.cs b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/PowerFx/Functions/MessageFunction.cs new file mode 100644 index 0000000000..e9d52c2e15 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/PowerFx/Functions/MessageFunction.cs @@ -0,0 +1,36 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Microsoft.Agents.AI.Workflows.Declarative.Extensions; +using Microsoft.Extensions.AI; +using Microsoft.PowerFx; +using Microsoft.PowerFx.Types; + +namespace Microsoft.Agents.AI.Workflows.Declarative.PowerFx.Functions; + +internal abstract class MessageFunction : ReflectionFunction +{ + protected MessageFunction(string functionName) + : base(functionName, FormulaType.String, FormulaType.String) + { } + + protected static FormulaValue Create(ChatRole role, StringValue input) => + string.IsNullOrEmpty(input.Value) ? + FormulaValue.NewBlank(RecordType.Empty()) : + FormulaValue.NewRecordFromFields( + new NamedValue(TypeSchema.Discriminator, nameof(ChatMessage).ToFormula()), + new NamedValue(TypeSchema.Message.Fields.Role, FormulaValue.New(role.Value)), + new NamedValue( + TypeSchema.Message.Fields.Content, + FormulaValue.NewTable( + RecordType.Empty() + .Add(TypeSchema.Message.Fields.ContentType, FormulaType.String) + .Add(TypeSchema.Message.Fields.ContentValue, FormulaType.String), + [ + FormulaValue.NewRecordFromFields( + new NamedValue(TypeSchema.Message.Fields.ContentType, FormulaValue.New(TypeSchema.Message.ContentTypes.Text)), + new NamedValue(TypeSchema.Message.Fields.ContentValue, input)) + ] + ) + ) + ); +} diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/PowerFx/Functions/MessageText.cs b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/PowerFx/Functions/MessageText.cs new file mode 100644 index 0000000000..ff9f7d499e --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/PowerFx/Functions/MessageText.cs @@ -0,0 +1,54 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Collections.Generic; +using Microsoft.PowerFx; +using Microsoft.PowerFx.Types; + +namespace Microsoft.Agents.AI.Workflows.Declarative.PowerFx.Functions; + +internal static class MessageText +{ + public const string FunctionName = nameof(MessageText); + + public sealed class StringInput() + : ReflectionFunction(FunctionName, FormulaType.String, FormulaType.String) + { + public static FormulaValue Execute(StringValue input) => input; + } + + public sealed class RecordInput() : ReflectionFunction(FunctionName, FormulaType.String, RecordType.Empty()) + { + public static FormulaValue Execute(RecordValue input) => FormulaValue.New(GetTextFromRecord(input)); + } + + public sealed class TableInput() : ReflectionFunction(FunctionName, FormulaType.String, TableType.Empty()) + { + public static FormulaValue Execute(TableValue tableValue) + { + return FormulaValue.New(string.Join("\n", GetText())); + + IEnumerable GetText() + { + foreach (DValue row in tableValue.Rows) + { + string text = GetTextFromRecord(row.Value); + if (!string.IsNullOrWhiteSpace(text)) + { + yield return text; + } + } + } + } + } + + private static string GetTextFromRecord(RecordValue recordValue) + { + FormulaValue textValue = recordValue.GetField(TypeSchema.Message.Fields.Text); + + return textValue switch + { + StringValue stringValue => stringValue.Value.Trim(), + _ => string.Empty, + }; + } +} diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/PowerFx/Functions/UserMessage.cs b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/PowerFx/Functions/UserMessage.cs index 1d4510b11d..968431623a 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/PowerFx/Functions/UserMessage.cs +++ b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/PowerFx/Functions/UserMessage.cs @@ -1,38 +1,15 @@ // Copyright (c) Microsoft. All rights reserved. -using Microsoft.Agents.AI.Workflows.Declarative.Extensions; using Microsoft.Extensions.AI; -using Microsoft.PowerFx; using Microsoft.PowerFx.Types; namespace Microsoft.Agents.AI.Workflows.Declarative.PowerFx.Functions; -internal sealed class UserMessage : ReflectionFunction +internal sealed class UserMessage : MessageFunction { public const string FunctionName = nameof(UserMessage); - public UserMessage() - : base(FunctionName, FormulaType.String, FormulaType.String) - { } + public UserMessage() : base(FunctionName) { } - public static FormulaValue Execute(StringValue input) => - string.IsNullOrEmpty(input.Value) ? - FormulaValue.NewBlank(RecordType.Empty()) : - FormulaValue.NewRecordFromFields( - new NamedValue(TypeSchema.Discriminator, nameof(ChatMessage).ToFormula()), - new NamedValue(TypeSchema.Message.Fields.Role, FormulaValue.New(ChatRole.User.Value)), - new NamedValue( - TypeSchema.Message.Fields.Content, - FormulaValue.NewTable( - RecordType.Empty() - .Add(TypeSchema.Message.Fields.ContentType, FormulaType.String) - .Add(TypeSchema.Message.Fields.ContentValue, FormulaType.String), - [ - FormulaValue.NewRecordFromFields( - new NamedValue(TypeSchema.Message.Fields.ContentType, FormulaValue.New(TypeSchema.Message.ContentTypes.Text)), - new NamedValue(TypeSchema.Message.Fields.ContentValue, input)) - ] - ) - ) - ); + public static FormulaValue Execute(StringValue input) => Create(ChatRole.User, input); } diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/PowerFx/RecalcEngineFactory.cs b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/PowerFx/RecalcEngineFactory.cs index 6d0364603c..2087307504 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/PowerFx/RecalcEngineFactory.cs +++ b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/PowerFx/RecalcEngineFactory.cs @@ -38,7 +38,11 @@ PowerFxConfig CreateConfig() } config.EnableSetFunction(); + config.AddFunction(new AgentMessage()); config.AddFunction(new UserMessage()); + config.AddFunction(new MessageText.StringInput()); + config.AddFunction(new MessageText.RecordInput()); + config.AddFunction(new MessageText.TableInput()); return config; } diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/README.md b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/README.md index 3a8fa6031d..4202b89b82 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/README.md +++ b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/README.md @@ -55,7 +55,7 @@ Please refer to the [README](../../samples/GettingStarted/Workflows/Declarative/ |**ConditionItem**|Represents a single conditional statement within a group. It evaluates a specific logical condition and determines the next step in the flow. |**ContinueLoop**|Skips the remaining steps in the current iteration and continues with the next loop cycle. Commonly used to bypass specific cases without exiting the loop entirely. |**EndConversation**|Terminates the current conversation session. It ensures any necessary cleanup or final actions are performed before closing. -|**EndDialog**|Ends the current dialog or sub-dialog within a broader conversation flow. This helps modularize complex interactions. +|**EndWorkflow**|Ends the current workflow or sub-workflow within a broader conversation flow. This helps modularize complex interactions. |**Foreach**|Iterates through a collection of items, executing a set of actions for each. Ideal for processing lists or batch operations. |**GotoAction**|Jumps directly to a specified action within the workflow. Enables non-linear navigation in the logic flow. diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/WorkflowAgentProvider.cs b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/WorkflowAgentProvider.cs index 1cb79db9f7..e0967ab376 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/WorkflowAgentProvider.cs +++ b/dotnet/src/Microsoft.Agents.AI.Workflows.Declarative/WorkflowAgentProvider.cs @@ -1,9 +1,11 @@ // Copyright (c) Microsoft. All rights reserved. using System.Collections.Generic; +using System.Text.Json.Nodes; using System.Threading; using System.Threading.Tasks; using Microsoft.Agents.AI.Workflows.Declarative.Events; +using Microsoft.Agents.AI.Workflows.Declarative.Extensions; using Microsoft.Extensions.AI; namespace Microsoft.Agents.AI.Workflows.Declarative; @@ -16,8 +18,8 @@ public abstract class WorkflowAgentProvider /// /// Gets or sets a collection of additional tools an agent is able to automatically invoke. /// If an agent is configured with a function tool that is not available, a is executed - /// that provides an that describes the function calls requested. The caller may - /// then respond with a corrsponding that includes the results of the function calls. + /// that provides an that describes the function calls requested. The caller may + /// then respond with a corrsponding that includes the results of the function calls. /// /// /// These will not impact the requests sent to the model by the . @@ -56,14 +58,6 @@ public abstract class WorkflowAgentProvider /// public bool AllowMultipleToolCalls { get; init; } - /// - /// Asynchronously retrieves an AI agent by its unique identifier. - /// - /// The unique identifier of the AI agent to retrieve. Cannot be null or empty. - /// A token that propagates notification when operation should be canceled. - /// The task result contains the associated. - public abstract Task GetAgentAsync(string agentId, CancellationToken cancellationToken = default); - /// /// Asynchronously creates a new conversation and returns its unique identifier. /// @@ -88,6 +82,24 @@ public abstract class WorkflowAgentProvider /// The requested message public abstract Task GetMessageAsync(string conversationId, string messageId, CancellationToken cancellationToken = default); + /// + /// Asynchronously retrieves an AI agent by its unique identifier. + /// + /// The unique identifier of the AI agent to retrieve. Cannot be null or empty. + /// An optional agent version. + /// Optional identifier of the target conversation. + /// The messages to include in the invocation. + /// Optional input arguments for agents that provide support. + /// A token that propagates notification when operation should be canceled. + /// Asynchronous set of . + public abstract IAsyncEnumerable InvokeAgentAsync( + string agentId, + string? agentVersion, + string? conversationId, + IEnumerable? messages, + IDictionary? inputArguments, + CancellationToken cancellationToken = default); + /// /// Retrieves a set of messages from a conversation. /// @@ -105,4 +117,14 @@ public abstract IAsyncEnumerable GetMessagesAsync( string? before = null, bool newestFirst = false, CancellationToken cancellationToken = default); + + /// + /// Utility method to convert a dictionary of input arguments to a JsonNode. + /// + /// The dictionary of input arguments. + /// A JsonNode representing the input arguments. + protected static JsonNode ConvertDictionaryToJson(IDictionary inputArguments) + { + return inputArguments.ToFormula().ToJson(); + } } diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows/AgentWorkflowBuilder.cs b/dotnet/src/Microsoft.Agents.AI.Workflows/AgentWorkflowBuilder.cs index 41f0d834f0..c5272e39ea 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows/AgentWorkflowBuilder.cs +++ b/dotnet/src/Microsoft.Agents.AI.Workflows/AgentWorkflowBuilder.cs @@ -139,7 +139,7 @@ private static Workflow BuildConcurrentCore( aggregator ??= static lists => (from list in lists where list.Count > 0 select list.Last()).ToList(); Func> endFactory = - (string _, string __) => new(new ConcurrentEndExecutor(agentExecutors.Length, aggregator)); + (_, __) => new(new ConcurrentEndExecutor(agentExecutors.Length, aggregator)); ExecutorBinding end = endFactory.BindExecutor(ConcurrentEndExecutor.ExecutorId); diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows/ChatForwardingExecutor.cs b/dotnet/src/Microsoft.Agents.AI.Workflows/ChatForwardingExecutor.cs new file mode 100644 index 0000000000..5bb2f5e237 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI.Workflows/ChatForwardingExecutor.cs @@ -0,0 +1,71 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Collections.Generic; +using System.Linq; +using System.Threading; +using System.Threading.Tasks; +using Microsoft.Extensions.AI; + +namespace Microsoft.Agents.AI.Workflows; + +/// +/// Provides configuration options for . +/// +public class ChatForwardingExecutorOptions +{ + /// + /// Gets or sets the chat role to use when converting string messages to instances. + /// If set, the executor will accept string messages and convert them to chat messages with this role. + /// + public ChatRole? StringMessageChatRole { get; set; } +} + +/// +/// A ChatProtocol executor that forwards all messages it receives. Useful for splitting inputs into parallel +/// processing paths. +/// +/// This executor is designed to be cross-run shareable and can be reset to its initial state. It handles +/// multiple chat-related types, enabling flexible message forwarding scenarios. Thread safety and reusability are +/// ensured by its design. +/// The unique identifier for the executor instance. Used to distinguish this executor within the system. +/// Optional configuration settings for the executor. If null, default options are used. +public sealed class ChatForwardingExecutor(string id, ChatForwardingExecutorOptions? options = null) : Executor(id, declareCrossRunShareable: true), IResettableExecutor +{ + private readonly ChatRole? _stringMessageChatRole = options?.StringMessageChatRole; + + /// + protected override RouteBuilder ConfigureRoutes(RouteBuilder routeBuilder) + { + if (this._stringMessageChatRole.HasValue) + { + routeBuilder = routeBuilder.AddHandler( + (message, context) => context.SendMessageAsync(new ChatMessage(ChatRole.User, message))); + } + + return routeBuilder.AddHandler(ForwardMessageAsync) + .AddHandler>(ForwardMessagesAsync) + .AddHandler(ForwardMessagesAsync) + .AddHandler>(ForwardMessagesAsync) + .AddHandler(ForwardTurnTokenAsync); + } + + private static ValueTask ForwardMessageAsync(ChatMessage message, IWorkflowContext context, CancellationToken cancellationToken) + => context.SendMessageAsync(message, cancellationToken); + + // Note that this can be used to split a turn into multiple parallel turns taken, which will cause streaming ChatMessages + // to overlap. + private static ValueTask ForwardTurnTokenAsync(TurnToken message, IWorkflowContext context, CancellationToken cancellationToken) + => context.SendMessageAsync(message, cancellationToken); + + // TODO: This is not ideal, but until we have a way of guaranteeing correct routing of interfaces across serialization + // boundaries, we need to do type unification. It behaves better when used as a handler in ChatProtocolExecutor because + // it is a strictly contravariant use, whereas this forces invariance on the type because it is directly forwarded. + private static ValueTask ForwardMessagesAsync(IEnumerable messages, IWorkflowContext context, CancellationToken cancellationToken) + => context.SendMessageAsync(messages is List messageList ? messageList : messages.ToList(), cancellationToken); + + private static ValueTask ForwardMessagesAsync(ChatMessage[] messages, IWorkflowContext context, CancellationToken cancellationToken) + => context.SendMessageAsync(messages, cancellationToken); + + /// + public ValueTask ResetAsync() => default; +} diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows/ChatProtocolExecutor.cs b/dotnet/src/Microsoft.Agents.AI.Workflows/ChatProtocolExecutor.cs index 56fb326338..238734b598 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows/ChatProtocolExecutor.cs +++ b/dotnet/src/Microsoft.Agents.AI.Workflows/ChatProtocolExecutor.cs @@ -26,7 +26,7 @@ public class ChatProtocolExecutorOptions /// public abstract class ChatProtocolExecutor : StatefulExecutor> { - private readonly static Func> s_initFunction = () => []; + private static readonly Func> s_initFunction = () => []; private readonly ChatRole? _stringMessageChatRole; /// diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows/Checkpointing/RepresentationExtensions.cs b/dotnet/src/Microsoft.Agents.AI.Workflows/Checkpointing/RepresentationExtensions.cs index 3d76c965bd..7c5b8183a6 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows/Checkpointing/RepresentationExtensions.cs +++ b/dotnet/src/Microsoft.Agents.AI.Workflows/Checkpointing/RepresentationExtensions.cs @@ -46,7 +46,7 @@ public static WorkflowInfo ToWorkflowInfo(this Workflow workflow) keySelector: sourceId => sourceId, elementSelector: sourceId => workflow.Edges[sourceId].Select(ToEdgeInfo).ToList()); - HashSet inputPorts = new(workflow.Ports.Values.Select(ToPortInfo)); + HashSet inputPorts = [.. workflow.Ports.Values.Select(ToPortInfo)]; return new WorkflowInfo(executors, edges, inputPorts, workflow.StartExecutorId, workflow.OutputExecutors); } diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows/DirectEdgeData.cs b/dotnet/src/Microsoft.Agents.AI.Workflows/DirectEdgeData.cs index 2119bd775b..7d61c939cd 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows/DirectEdgeData.cs +++ b/dotnet/src/Microsoft.Agents.AI.Workflows/DirectEdgeData.cs @@ -11,7 +11,7 @@ namespace Microsoft.Agents.AI.Workflows; /// public sealed class DirectEdgeData : EdgeData { - internal DirectEdgeData(string sourceId, string sinkId, EdgeId id, PredicateT? condition = null) : base(id) + internal DirectEdgeData(string sourceId, string sinkId, EdgeId id, PredicateT? condition = null, string? label = null) : base(id, label) { this.SourceId = sourceId; this.SinkId = sinkId; diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows/EdgeData.cs b/dotnet/src/Microsoft.Agents.AI.Workflows/EdgeData.cs index 7771b3966e..570bc79bc0 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows/EdgeData.cs +++ b/dotnet/src/Microsoft.Agents.AI.Workflows/EdgeData.cs @@ -14,10 +14,16 @@ public abstract class EdgeData /// internal abstract EdgeConnection Connection { get; } - internal EdgeData(EdgeId id) + internal EdgeData(EdgeId id, string? label = null) { this.Id = id; + this.Label = label; } internal EdgeId Id { get; } + + /// + /// An optional label for the edge, allowing for arbitrary metadata to be associated with it. + /// + public string? Label { get; } } diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows/Execution/AsyncRunHandleExtensions.cs b/dotnet/src/Microsoft.Agents.AI.Workflows/Execution/AsyncRunHandleExtensions.cs index c7ac339a0c..c9936ce683 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows/Execution/AsyncRunHandleExtensions.cs +++ b/dotnet/src/Microsoft.Agents.AI.Workflows/Execution/AsyncRunHandleExtensions.cs @@ -8,7 +8,7 @@ namespace Microsoft.Agents.AI.Workflows.Execution; internal static class AsyncRunHandleExtensions { - public async static ValueTask> WithCheckpointingAsync(this AsyncRunHandle runHandle, Func> prepareFunc) + public static async ValueTask> WithCheckpointingAsync(this AsyncRunHandle runHandle, Func> prepareFunc) { TRunType run = await prepareFunc().ConfigureAwait(false); return new Checkpointed(run, runHandle); diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows/Execution/EdgeConnection.cs b/dotnet/src/Microsoft.Agents.AI.Workflows/Execution/EdgeConnection.cs index 8833907489..6780b26fe7 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows/Execution/EdgeConnection.cs +++ b/dotnet/src/Microsoft.Agents.AI.Workflows/Execution/EdgeConnection.cs @@ -41,8 +41,8 @@ public EdgeConnection(List sourceIds, List sinkIds) /// contains duplicate values. public static EdgeConnection CreateChecked(List sourceIds, List sinkIds) { - HashSet sourceSet = new(Throw.IfNull(sourceIds)); - HashSet sinkSet = new(Throw.IfNull(sinkIds)); + HashSet sourceSet = [.. Throw.IfNull(sourceIds)]; + HashSet sinkSet = [.. Throw.IfNull(sinkIds)]; if (sourceSet.Count != sourceIds.Count) { diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows/Execution/FanInEdgeState.cs b/dotnet/src/Microsoft.Agents.AI.Workflows/Execution/FanInEdgeState.cs index fe564f1c38..9c6a941a11 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows/Execution/FanInEdgeState.cs +++ b/dotnet/src/Microsoft.Agents.AI.Workflows/Execution/FanInEdgeState.cs @@ -14,7 +14,7 @@ internal sealed class FanInEdgeState public FanInEdgeState(FanInEdgeData fanInEdge) { this.SourceIds = fanInEdge.SourceIds.ToArray(); - this.Unseen = new(this.SourceIds); + this.Unseen = [.. this.SourceIds]; this._pendingMessages = []; } @@ -40,7 +40,7 @@ public FanInEdgeState(string[] sourceIds, HashSet unseen, List takenMessages = Interlocked.Exchange(ref this._pendingMessages, []); - this.Unseen = new(this.SourceIds); + this.Unseen = [.. this.SourceIds]; if (takenMessages.Count == 0) { diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows/Execution/NonThrowingChannelReaderAsyncEnumerable.cs b/dotnet/src/Microsoft.Agents.AI.Workflows/Execution/NonThrowingChannelReaderAsyncEnumerable.cs index aaae42f2f1..306373f4b7 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows/Execution/NonThrowingChannelReaderAsyncEnumerable.cs +++ b/dotnet/src/Microsoft.Agents.AI.Workflows/Execution/NonThrowingChannelReaderAsyncEnumerable.cs @@ -16,8 +16,7 @@ internal sealed class NonThrowingChannelReaderAsyncEnumerable(ChannelReader reader, CancellationToken cancellationToken) : IAsyncEnumerator { - private T? _current; - public T Current => this._current ?? throw new InvalidOperationException("Enumeration not started."); + public T Current { get => field ?? throw new InvalidOperationException("Enumeration not started."); private set; } public ValueTask DisposeAsync() { @@ -36,7 +35,7 @@ public async ValueTask MoveNextAsync() bool hasData = await reader.WaitToReadAsync(cancellationToken).ConfigureAwait(false); if (hasData) { - this._current = await reader.ReadAsync(cancellationToken).ConfigureAwait(false); + this.Current = await reader.ReadAsync(cancellationToken).ConfigureAwait(false); return true; } } diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows/Execution/StateScope.cs b/dotnet/src/Microsoft.Agents.AI.Workflows/Execution/StateScope.cs index e1c50ab1a3..93960f0f9a 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows/Execution/StateScope.cs +++ b/dotnet/src/Microsoft.Agents.AI.Workflows/Execution/StateScope.cs @@ -51,7 +51,7 @@ public bool ContainsKey(string key) Throw.IfNullOrEmpty(key); if (this._stateData.TryGetValue(key, out PortableValue? value)) { - if (typeof(T) == typeof(PortableValue) && !value.TypeId.IsMatch(typeof(PortableValue))) + if (typeof(T) == typeof(PortableValue) && !value.TypeId.IsMatch()) { // value is PortableValue, and we do not need to unwrap a PortableValue instance inside of it // Unfortunately we need to cast through object here. diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows/Executor.cs b/dotnet/src/Microsoft.Agents.AI.Workflows/Executor.cs index e0b53429f9..647dbcd852 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows/Executor.cs +++ b/dotnet/src/Microsoft.Agents.AI.Workflows/Executor.cs @@ -93,18 +93,17 @@ protected virtual ISet ConfigureYieldTypes() return new HashSet(); } - private MessageRouter? _router; internal MessageRouter Router { get { - if (this._router is null) + if (field is null) { RouteBuilder routeBuilder = this.ConfigureRoutes(new RouteBuilder()); - this._router = routeBuilder.Build(); + field = routeBuilder.Build(); } - return this._router; + return field; } } diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows/FanInEdgeData.cs b/dotnet/src/Microsoft.Agents.AI.Workflows/FanInEdgeData.cs index 0cb2b38378..1132fca334 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows/FanInEdgeData.cs +++ b/dotnet/src/Microsoft.Agents.AI.Workflows/FanInEdgeData.cs @@ -10,7 +10,7 @@ namespace Microsoft.Agents.AI.Workflows; /// internal sealed class FanInEdgeData : EdgeData { - internal FanInEdgeData(List sourceIds, string sinkId, EdgeId id) : base(id) + internal FanInEdgeData(List sourceIds, string sinkId, EdgeId id, string? label) : base(id, label) { this.SourceIds = sourceIds; this.SinkId = sinkId; diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows/FanOutEdgeData.cs b/dotnet/src/Microsoft.Agents.AI.Workflows/FanOutEdgeData.cs index 9d9ddf4cea..86a940c1b6 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows/FanOutEdgeData.cs +++ b/dotnet/src/Microsoft.Agents.AI.Workflows/FanOutEdgeData.cs @@ -13,7 +13,7 @@ namespace Microsoft.Agents.AI.Workflows; /// internal sealed class FanOutEdgeData : EdgeData { - internal FanOutEdgeData(string sourceId, List sinkIds, EdgeId edgeId, AssignerF? assigner = null) : base(edgeId) + internal FanOutEdgeData(string sourceId, List sinkIds, EdgeId edgeId, AssignerF? assigner = null, string? label = null) : base(edgeId, label) { this.SourceId = sourceId; this.SinkIds = sinkIds; diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows/GroupChatManager.cs b/dotnet/src/Microsoft.Agents.AI.Workflows/GroupChatManager.cs index 9d3d55b33f..d16a4b5b43 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows/GroupChatManager.cs +++ b/dotnet/src/Microsoft.Agents.AI.Workflows/GroupChatManager.cs @@ -13,8 +13,6 @@ namespace Microsoft.Agents.AI.Workflows; /// public abstract class GroupChatManager { - private int _maximumIterationCount = 40; - /// /// Initializes a new instance of the class. /// @@ -34,9 +32,9 @@ protected GroupChatManager() { } /// public int MaximumIterationCount { - get => this._maximumIterationCount; - set => this._maximumIterationCount = Throw.IfLessThan(value, 1); - } + get; + set => field = Throw.IfLessThan(value, 1); + } = 40; /// /// Selects the next agent to participate in the group chat based on the provided chat history and team. diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows/GroupChatWorkflowBuilder.cs b/dotnet/src/Microsoft.Agents.AI.Workflows/GroupChatWorkflowBuilder.cs index c02a609f75..12b0f9c707 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows/GroupChatWorkflowBuilder.cs +++ b/dotnet/src/Microsoft.Agents.AI.Workflows/GroupChatWorkflowBuilder.cs @@ -53,7 +53,7 @@ public Workflow Build() Dictionary agentMap = agents.ToDictionary(a => a, a => (ExecutorBinding)new AgentRunStreamingExecutor(a, includeInputInOutput: true)); Func> groupChatHostFactory = - (string id, string runId) => new(new GroupChatHost(id, agents, agentMap, this._managerFactory)); + (id, runId) => new(new GroupChatHost(id, agents, agentMap, this._managerFactory)); ExecutorBinding host = groupChatHostFactory.BindExecutor(nameof(GroupChatHost)); WorkflowBuilder builder = new(host); diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows/InProc/InProcessRunner.cs b/dotnet/src/Microsoft.Agents.AI.Workflows/InProc/InProcessRunner.cs index 9c100ecbbf..8c7149b0be 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows/InProc/InProcessRunner.cs +++ b/dotnet/src/Microsoft.Agents.AI.Workflows/InProc/InProcessRunner.cs @@ -225,7 +225,7 @@ private async ValueTask RunSuperstepAsync(StepContext currentStep, CancellationT // subworkflow's input queue. In order to actually process the message and align the supersteps correctly, // we need to drive the superstep of the subworkflow here. // TODO: Investigate if we can fully pull in the subworkflow execution into the WorkflowHostExecutor itself. - List subworkflowTasks = new(); + List subworkflowTasks = []; foreach (ISuperStepRunner subworkflowRunner in this.RunContext.JoinedSubworkflowRunners) { subworkflowTasks.Add(subworkflowRunner.RunSuperStepAsync(cancellationToken).AsTask()); diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows/Microsoft.Agents.AI.Workflows.csproj b/dotnet/src/Microsoft.Agents.AI.Workflows/Microsoft.Agents.AI.Workflows.csproj index ff2e9dee64..7379d9a6ac 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows/Microsoft.Agents.AI.Workflows.csproj +++ b/dotnet/src/Microsoft.Agents.AI.Workflows/Microsoft.Agents.AI.Workflows.csproj @@ -1,8 +1,6 @@ - $(ProjectsTargetFrameworks) - $(ProjectsDebugTargetFrameworks) preview diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows/Reflection/RouteBuilderExtensions.cs b/dotnet/src/Microsoft.Agents.AI.Workflows/Reflection/RouteBuilderExtensions.cs index c0c6b8c8ca..f25f896db9 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows/Reflection/RouteBuilderExtensions.cs +++ b/dotnet/src/Microsoft.Agents.AI.Workflows/Reflection/RouteBuilderExtensions.cs @@ -11,7 +11,7 @@ namespace Microsoft.Agents.AI.Workflows.Reflection; internal static class IMessageHandlerReflection { - private const string Nameof_HandleAsync = nameof(IMessageHandler.HandleAsync); + private const string Nameof_HandleAsync = nameof(IMessageHandler<>.HandleAsync); internal static readonly MethodInfo HandleAsync_1 = typeof(IMessageHandler<>).GetMethod(Nameof_HandleAsync, BindingFlags.Public | BindingFlags.Instance)!; internal static readonly MethodInfo HandleAsync_2 = typeof(IMessageHandler<,>).GetMethod(Nameof_HandleAsync, BindingFlags.Public | BindingFlags.Instance)!; diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows/Reflection/ValueTaskTypeErasure.cs b/dotnet/src/Microsoft.Agents.AI.Workflows/Reflection/ValueTaskTypeErasure.cs index f8aa22b8b6..90e184c30e 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows/Reflection/ValueTaskTypeErasure.cs +++ b/dotnet/src/Microsoft.Agents.AI.Workflows/Reflection/ValueTaskTypeErasure.cs @@ -9,7 +9,7 @@ namespace Microsoft.Agents.AI.Workflows.Reflection; internal static class ValueTaskReflection { - private const string Nameof_AsTask = nameof(ValueTask.AsTask); + private const string Nameof_AsTask = nameof(ValueTask<>.AsTask); internal static readonly MethodInfo AsTask = typeof(ValueTask<>).GetMethod(Nameof_AsTask, BindingFlags.Public | BindingFlags.Instance)!; internal static MethodInfo ReflectAsTask(this Type specializedType) @@ -25,7 +25,7 @@ internal static MethodInfo ReflectAsTask(this Type specializedType) internal static class TaskReflection { - private const string Nameof_Result = nameof(Task.Result); + private const string Nameof_Result = nameof(Task<>.Result); internal static readonly MethodInfo Result_get = typeof(Task<>).GetProperty(Nameof_Result)!.GetMethod!; internal static MethodInfo ReflectResult_get(this Type specializedType) diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows/Specialized/ChatForwardingExecutor.cs b/dotnet/src/Microsoft.Agents.AI.Workflows/Specialized/ChatForwardingExecutor.cs deleted file mode 100644 index b395dd4216..0000000000 --- a/dotnet/src/Microsoft.Agents.AI.Workflows/Specialized/ChatForwardingExecutor.cs +++ /dev/null @@ -1,20 +0,0 @@ -// Copyright (c) Microsoft. All rights reserved. - -using System.Collections.Generic; -using System.Threading.Tasks; -using Microsoft.Extensions.AI; - -namespace Microsoft.Agents.AI.Workflows.Specialized; - -/// Executor that forwards all messages. -internal sealed class ChatForwardingExecutor(string id) : Executor(id, declareCrossRunShareable: true), IResettableExecutor -{ - protected override RouteBuilder ConfigureRoutes(RouteBuilder routeBuilder) => - routeBuilder - .AddHandler((message, context, cancellationToken) => context.SendMessageAsync(new ChatMessage(ChatRole.User, message), cancellationToken: cancellationToken)) - .AddHandler((message, context, cancellationToken) => context.SendMessageAsync(message, cancellationToken: cancellationToken)) - .AddHandler>((messages, context, cancellationToken) => context.SendMessageAsync(messages, cancellationToken: cancellationToken)) - .AddHandler((turnToken, context, cancellationToken) => context.SendMessageAsync(turnToken, cancellationToken: cancellationToken)); - - public ValueTask ResetAsync() => default; -} diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows/Specialized/RequestInfoExecutor.cs b/dotnet/src/Microsoft.Agents.AI.Workflows/Specialized/RequestInfoExecutor.cs index afb07507f9..932cf297a3 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows/Specialized/RequestInfoExecutor.cs +++ b/dotnet/src/Microsoft.Agents.AI.Workflows/Specialized/RequestInfoExecutor.cs @@ -10,13 +10,11 @@ namespace Microsoft.Agents.AI.Workflows.Specialized; -internal sealed class RequestPortOptions -{ -} +internal sealed class RequestPortOptions; internal sealed class RequestInfoExecutor : Executor { - private readonly Dictionary _wrappedRequests = new(); + private readonly Dictionary _wrappedRequests = []; private RequestPort Port { get; } private IExternalRequestSink? RequestSink { get; set; } diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows/StatefulExecutor.cs b/dotnet/src/Microsoft.Agents.AI.Workflows/StatefulExecutor.cs index 344134369d..12079289a4 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows/StatefulExecutor.cs +++ b/dotnet/src/Microsoft.Agents.AI.Workflows/StatefulExecutor.cs @@ -110,7 +110,7 @@ protected async ValueTask InvokeWithStateAsync( { if (!skipCache && !context.ConcurrentRunsEnabled) { - TState newState = await invocation(this._stateCache ?? (this._initialStateFactory()), + TState newState = await invocation(this._stateCache ?? this._initialStateFactory(), context, cancellationToken).ConfigureAwait(false) ?? this._initialStateFactory(); diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows/Visualization/WorkflowVisualizer.cs b/dotnet/src/Microsoft.Agents.AI.Workflows/Visualization/WorkflowVisualizer.cs index ebf6f08ffb..e1b69e9f9e 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows/Visualization/WorkflowVisualizer.cs +++ b/dotnet/src/Microsoft.Agents.AI.Workflows/Visualization/WorkflowVisualizer.cs @@ -99,10 +99,30 @@ private static void EmitWorkflowDigraph(Workflow workflow, List lines, s } // Emit normal edges - foreach (var (src, target, isConditional) in ComputeNormalEdges(workflow)) + foreach (var (src, target, isConditional, label) in ComputeNormalEdges(workflow)) { - var edgeAttr = isConditional ? " [style=dashed, label=\"conditional\"]" : ""; - lines.Add($"{indent}\"{MapId(src)}\" -> \"{MapId(target)}\"{edgeAttr};"); + // Build edge attributes + var attributes = new List(); + + // Add style for conditional edges + if (isConditional) + { + attributes.Add("style=dashed"); + } + + // Add label (custom label or default "conditional" for conditional edges) + if (label != null) + { + attributes.Add($"label=\"{EscapeDotLabel(label)}\""); + } + else if (isConditional) + { + attributes.Add("label=\"conditional\""); + } + + // Combine attributes + var attrString = attributes.Count > 0 ? $" [{string.Join(", ", attributes)}]" : ""; + lines.Add($"{indent}\"{MapId(src)}\" -> \"{MapId(target)}\"{attrString};"); } } @@ -133,12 +153,7 @@ private static void EmitSubWorkflowsDigraph(Workflow workflow, List line private static void EmitWorkflowMermaid(Workflow workflow, List lines, string indent, string? ns = null) { - string sanitize(string input) - { - return input; - } - - string MapId(string id) => ns != null ? $"{sanitize(ns)}/{sanitize(id)}" : id; + string MapId(string id) => ns != null ? $"{ns}/{id}" : id; // Add start node var startExecutorId = workflow.StartExecutorId; @@ -175,14 +190,23 @@ string sanitize(string input) } // Emit normal edges - foreach (var (src, target, isConditional) in ComputeNormalEdges(workflow)) + foreach (var (src, target, isConditional, label) in ComputeNormalEdges(workflow)) { if (isConditional) { - lines.Add($"{indent}{MapId(src)} -. conditional .--> {MapId(target)};"); + string effectiveLabel = label != null ? EscapeMermaidLabel(label) : "conditional"; + + // Conditional edge, with user label or default + lines.Add($"{indent}{MapId(src)} -. {effectiveLabel} .--> {MapId(target)};"); + } + else if (label != null) + { + // Regular edge with label + lines.Add($"{indent}{MapId(src)} -->|{EscapeMermaidLabel(label)}| {MapId(target)};"); } else { + // Regular edge without label lines.Add($"{indent}{MapId(src)} --> {MapId(target)};"); } } @@ -214,9 +238,9 @@ string sanitize(string input) return result; } - private static List<(string Source, string Target, bool IsConditional)> ComputeNormalEdges(Workflow workflow) + private static List<(string Source, string Target, bool IsConditional, string? Label)> ComputeNormalEdges(Workflow workflow) { - var edges = new List<(string, string, bool)>(); + var edges = new List<(string, string, bool, string?)>(); foreach (var edgeGroup in workflow.Edges.Values.SelectMany(x => x)) { if (edgeGroup.Kind == EdgeKind.FanIn) @@ -229,14 +253,15 @@ string sanitize(string input) case EdgeKind.Direct when edgeGroup.DirectEdgeData != null: var directData = edgeGroup.DirectEdgeData; var isConditional = directData.Condition != null; - edges.Add((directData.SourceId, directData.SinkId, isConditional)); + var label = directData.Label; + edges.Add((directData.SourceId, directData.SinkId, isConditional, label)); break; case EdgeKind.FanOut when edgeGroup.FanOutEdgeData != null: var fanOutData = edgeGroup.FanOutEdgeData; foreach (var sinkId in fanOutData.SinkIds) { - edges.Add((fanOutData.SourceId, sinkId, false)); + edges.Add((fanOutData.SourceId, sinkId, false, fanOutData.Label)); } break; } @@ -276,5 +301,24 @@ private static bool TryGetNestedWorkflow(ExecutorBinding binding, [NotNullWhen(t return false; } + // Helper method to escape special characters in DOT labels + private static string EscapeDotLabel(string label) + { + return label.Replace("\"", "\\\"").Replace("\n", "\\n"); + } + + // Helper method to escape special characters in Mermaid labels + private static string EscapeMermaidLabel(string label) + { + return label + .Replace("&", "&") // Must be first to avoid double-escaping + .Replace("|", "|") // Pipe breaks Mermaid delimiter syntax + .Replace("\"", """) // Quote character + .Replace("<", "<") // Less than + .Replace(">", ">") // Greater than + .Replace("\n", "
") // Newline to HTML break + .Replace("\r", ""); // Remove carriage return + } + #endregion } diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows/Workflow.cs b/dotnet/src/Microsoft.Agents.AI.Workflows/Workflow.cs index a4f6be1210..456838b9eb 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows/Workflow.cs +++ b/dotnet/src/Microsoft.Agents.AI.Workflows/Workflow.cs @@ -120,7 +120,7 @@ internal void CheckOwnership(object? existingOwnershipSignoff = null) throw new InvalidOperationException($"Existing ownership does not match check value. {Summarize(maybeOwned)} vs. {Summarize(existingOwnershipSignoff)}"); } - string Summarize(object? maybeOwnerToken) => maybeOwnerToken switch + static string Summarize(object? maybeOwnerToken) => maybeOwnerToken switch { string s => $"'{s}'", null => "", @@ -168,11 +168,8 @@ internal void TakeOwnership(object ownerToken, bool subworkflow = false, object? Justification = "Does not exist in NetFx 4.7.2")] internal async ValueTask ReleaseOwnershipAsync(object ownerToken) { - object? originalToken = Interlocked.CompareExchange(ref this._ownerToken, null, ownerToken); - if (originalToken == null) - { + object? originalToken = Interlocked.CompareExchange(ref this._ownerToken, null, ownerToken) ?? throw new InvalidOperationException("Attempting to release ownership of a Workflow that is not owned."); - } if (!ReferenceEquals(originalToken, ownerToken)) { diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows/WorkflowBuilder.cs b/dotnet/src/Microsoft.Agents.AI.Workflows/WorkflowBuilder.cs index 674a65f7f6..4b6980d433 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows/WorkflowBuilder.cs +++ b/dotnet/src/Microsoft.Agents.AI.Workflows/WorkflowBuilder.cs @@ -168,6 +168,18 @@ private HashSet EnsureEdgesFor(string sourceId) return edges; } + /// + /// Adds a directed edge from the specified source executor to the target executor, optionally guarded by a + /// condition. + /// + /// The executor that acts as the source node of the edge. Cannot be null. + /// The executor that acts as the target node of the edge. Cannot be null. + /// The current instance of . + /// Thrown if an unconditional edge between the specified source and target + /// executors already exists. + public WorkflowBuilder AddEdge(ExecutorBinding source, ExecutorBinding target) + => this.AddEdge(source, target, null, false); + /// /// Adds a directed edge from the specified source executor to the target executor, optionally guarded by a /// condition. @@ -182,6 +194,20 @@ private HashSet EnsureEdgesFor(string sourceId) public WorkflowBuilder AddEdge(ExecutorBinding source, ExecutorBinding target, bool idempotent = false) => this.AddEdge(source, target, null, idempotent); + /// + /// Adds a directed edge from the specified source executor to the target executor. + /// + /// The executor that acts as the source node of the edge. Cannot be null. + /// The executor that acts as the target node of the edge. Cannot be null. + /// An optional label for the edge. Will be used in visualizations. + /// If set to , adding the same edge multiple times will be a NoOp, + /// rather than an error. + /// The current instance of . + /// Thrown if an unconditional edge between the specified source and target + /// executors already exists. + public WorkflowBuilder AddEdge(ExecutorBinding source, ExecutorBinding target, string? label = null, bool idempotent = false) + => this.AddEdge(source, target, null, label, idempotent); + internal static Func? CreateConditionFunc(Func? condition) { if (condition is null) @@ -222,6 +248,20 @@ public WorkflowBuilder AddEdge(ExecutorBinding source, ExecutorBinding target, b private EdgeId TakeEdgeId() => new(Interlocked.Increment(ref this._edgeCount)); + /// + /// Adds a directed edge from the specified source executor to the target executor, optionally guarded by a + /// condition. + /// + /// The executor that acts as the source node of the edge. Cannot be null. + /// The executor that acts as the target node of the edge. Cannot be null. + /// An optional predicate that determines whether the edge should be followed based on the input. + /// If null, the edge is always activated when the source sends a message. + /// The current instance of . + /// Thrown if an unconditional edge between the specified source and target + /// executors already exists. + public WorkflowBuilder AddEdge(ExecutorBinding source, ExecutorBinding target, Func? condition = null) + => this.AddEdge(source, target, condition, label: null, false); + /// /// Adds a directed edge from the specified source executor to the target executor, optionally guarded by a /// condition. @@ -236,6 +276,23 @@ public WorkflowBuilder AddEdge(ExecutorBinding source, ExecutorBinding target, b /// Thrown if an unconditional edge between the specified source and target /// executors already exists. public WorkflowBuilder AddEdge(ExecutorBinding source, ExecutorBinding target, Func? condition = null, bool idempotent = false) + => this.AddEdge(source, target, condition, label: null, idempotent); + + /// + /// Adds a directed edge from the specified source executor to the target executor, optionally guarded by a + /// condition. + /// + /// The executor that acts as the source node of the edge. Cannot be null. + /// The executor that acts as the target node of the edge. Cannot be null. + /// An optional predicate that determines whether the edge should be followed based on the input. + /// An optional label for the edge. Will be used in visualizations. + /// If set to , adding the same edge multiple times will be a NoOp, + /// rather than an error. + /// If null, the edge is always activated when the source sends a message. + /// The current instance of . + /// Thrown if an unconditional edge between the specified source and target + /// executors already exists. + public WorkflowBuilder AddEdge(ExecutorBinding source, ExecutorBinding target, Func? condition = null, string? label = null, bool idempotent = false) { // Add an edge from source to target with an optional condition. // This is a low-level builder method that does not enforce any specific executor type. @@ -256,7 +313,7 @@ public WorkflowBuilder AddEdge(ExecutorBinding source, ExecutorBinding target "You cannot add another edge without a condition for the same source and target."); } - DirectEdgeData directEdge = new(this.Track(source).Id, this.Track(target).Id, this.TakeEdgeId(), CreateConditionFunc(condition)); + DirectEdgeData directEdge = new(this.Track(source).Id, this.Track(target).Id, this.TakeEdgeId(), CreateConditionFunc(condition), label); this.EnsureEdgesFor(source.Id).Add(new(directEdge)); @@ -275,6 +332,19 @@ public WorkflowBuilder AddEdge(ExecutorBinding source, ExecutorBinding target public WorkflowBuilder AddFanOutEdge(ExecutorBinding source, IEnumerable targets) => this.AddFanOutEdge(source, targets, null); + /// + /// Adds a fan-out edge from the specified source executor to one or more target executors, optionally using a + /// custom partitioning function. + /// + /// If a partitioner function is provided, it will be used to distribute input across the target + /// executors. The order of targets determines their mapping in the partitioning process. + /// The source executor from which the fan-out edge originates. Cannot be null. + /// One or more target executors that will receive the fan-out edge. Cannot be null or empty. + /// A label for the edge. Will be used in visualization. + /// The current instance of . + public WorkflowBuilder AddFanOutEdge(ExecutorBinding source, IEnumerable targets, string label) + => this.AddFanOutEdge(source, targets, null, label); + internal static Func>? CreateTargetAssignerFunc(Func>? targetAssigner) { if (targetAssigner is null) @@ -305,6 +375,21 @@ public WorkflowBuilder AddFanOutEdge(ExecutorBinding source, IEnumerableAn optional function that determines how input is assigned among the target executors. /// If null, messages will route to all targets. public WorkflowBuilder AddFanOutEdge(ExecutorBinding source, IEnumerable targets, Func>? targetSelector = null) + => this.AddFanOutEdge(source, targets, targetSelector, label: null); + + /// + /// Adds a fan-out edge from the specified source executor to one or more target executors, optionally using a + /// custom partitioning function. + /// + /// If a partitioner function is provided, it will be used to distribute input across the target + /// executors. The order of targets determines their mapping in the partitioning process. + /// The source executor from which the fan-out edge originates. Cannot be null. + /// One or more target executors that will receive the fan-out edge. Cannot be null or empty. + /// The current instance of . + /// An optional function that determines how input is assigned among the target executors. + /// If null, messages will route to all targets. + /// An optional label for the edge. Will be used in visualizations. + public WorkflowBuilder AddFanOutEdge(ExecutorBinding source, IEnumerable targets, Func>? targetSelector = null, string? label = null) { Throw.IfNull(source); Throw.IfNull(targets); @@ -321,7 +406,8 @@ public WorkflowBuilder AddFanOutEdge(ExecutorBinding source, IEnumerable(ExecutorBinding source, IEnumerableThe target executor that receives input from the specified source executors. Cannot be null. /// The current instance of . public WorkflowBuilder AddFanInEdge(IEnumerable sources, ExecutorBinding target) + => this.AddFanInEdge(sources, target, label: null); + + /// + /// Adds a fan-in edge to the workflow, connecting multiple source executors to a single target executor with an + /// optional trigger condition. + /// + /// This method establishes a fan-in relationship, allowing the target executor to be activated + /// based on the completion or state of multiple sources. The trigger parameter can be used to customize activation + /// behavior. + /// One or more source executors that provide input to the target. Cannot be null or empty. + /// The target executor that receives input from the specified source executors. Cannot be null. + /// An optional label for the edge. Will be used in visualizations. + /// The current instance of . + public WorkflowBuilder AddFanInEdge(IEnumerable sources, ExecutorBinding target, string? label = null) { Throw.IfNull(target); Throw.IfNull(sources); @@ -354,7 +454,8 @@ public WorkflowBuilder AddFanInEdge(IEnumerable sources, Execut FanInEdgeData edgeData = new( sourceIds, this.Track(target).Id, - this.TakeEdgeId()); + this.TakeEdgeId(), + label); foreach (string sourceId in edgeData.SourceIds) { @@ -380,7 +481,7 @@ private void Validate(bool validateOrphans) } // Make sure that all nodes are connected to the start executor (transitively) - HashSet remainingExecutors = new(this._executorBindings.Keys); + HashSet remainingExecutors = [.. this._executorBindings.Keys]; Queue toVisit = new([this._startExecutorId]); if (!validateOrphans) diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows/WorkflowHostAgent.cs b/dotnet/src/Microsoft.Agents.AI.Workflows/WorkflowHostAgent.cs index 98dc5903bf..70fcee15df 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows/WorkflowHostAgent.cs +++ b/dotnet/src/Microsoft.Agents.AI.Workflows/WorkflowHostAgent.cs @@ -39,7 +39,7 @@ public WorkflowHostAgent(Workflow workflow, string? id = null, string? name = nu this._describeTask = this._workflow.DescribeProtocolAsync().AsTask(); } - public override string Id => this._id ?? base.Id; + protected override string? IdCore => this._id; public override string? Name { get; } public override string? Description { get; } diff --git a/dotnet/src/Microsoft.Agents.AI.Workflows/WorkflowThread.cs b/dotnet/src/Microsoft.Agents.AI.Workflows/WorkflowThread.cs index ffa044791f..d27de6bd5c 100644 --- a/dotnet/src/Microsoft.Agents.AI.Workflows/WorkflowThread.cs +++ b/dotnet/src/Microsoft.Agents.AI.Workflows/WorkflowThread.cs @@ -68,9 +68,6 @@ public WorkflowThread(Workflow workflow, JsonElement serializedThread, IWorkflow public CheckpointInfo? LastCheckpoint { get; set; } - protected override Task MessagesReceivedAsync(IEnumerable newMessages, CancellationToken cancellationToken = default) - => this.MessageStore.AddMessagesAsync(newMessages, cancellationToken); - public override JsonElement Serialize(JsonSerializerOptions? jsonSerializerOptions = null) { JsonMarshaller marshaller = new(jsonSerializerOptions); diff --git a/dotnet/src/Microsoft.Agents.AI/AgentJsonUtilities.cs b/dotnet/src/Microsoft.Agents.AI/AgentJsonUtilities.cs index 36bef4a2af..fe3f73b28b 100644 --- a/dotnet/src/Microsoft.Agents.AI/AgentJsonUtilities.cs +++ b/dotnet/src/Microsoft.Agents.AI/AgentJsonUtilities.cs @@ -4,7 +4,6 @@ using System.Text.Encodings.Web; using System.Text.Json; using System.Text.Json.Serialization; -using Microsoft.Agents.AI.Data; namespace Microsoft.Agents.AI; @@ -68,6 +67,7 @@ private static JsonSerializerOptions CreateDefaultOptions() // Agent abstraction types [JsonSerializable(typeof(ChatClientAgentThread.ThreadState))] [JsonSerializable(typeof(TextSearchProvider.TextSearchProviderState))] + [JsonSerializable(typeof(ChatHistoryMemoryProvider.ChatHistoryMemoryProviderState))] [ExcludeFromCodeCoverage] internal sealed partial class JsonContext : JsonSerializerContext; diff --git a/dotnet/src/Microsoft.Agents.AI/ChatClient/ChatClientAgent.cs b/dotnet/src/Microsoft.Agents.AI/ChatClient/ChatClientAgent.cs index 46f893e531..a5a34d24a9 100644 --- a/dotnet/src/Microsoft.Agents.AI/ChatClient/ChatClientAgent.cs +++ b/dotnet/src/Microsoft.Agents.AI/ChatClient/ChatClientAgent.cs @@ -59,13 +59,13 @@ public ChatClientAgent(IChatClient chatClient, string? instructions = null, stri chatClient, new ChatClientAgentOptions { - Name = name, - Description = description, - Instructions = instructions, - ChatOptions = tools is null ? null : new ChatOptions + ChatOptions = (tools is null && string.IsNullOrWhiteSpace(instructions)) ? null : new ChatOptions { Tools = tools, - } + Instructions = instructions + }, + Name = name, + Description = description }, loggerFactory, services) @@ -121,7 +121,7 @@ public ChatClientAgent(IChatClient chatClient, ChatClientAgentOptions? options, public IChatClient ChatClient { get; } /// - public override string Id => this._agentOptions?.Id ?? base.Id; + protected override string? IdCore => this._agentOptions?.Id; /// public override string? Name => this._agentOptions?.Name; @@ -141,7 +141,7 @@ public ChatClientAgent(IChatClient chatClient, ChatClientAgentOptions? options, /// These instructions are typically provided to the AI model as system messages to establish /// the context and expected behavior for the agent's responses. /// - public string? Instructions => this._agentOptions?.Instructions; + public string? Instructions => this._agentOptions?.ChatOptions?.Instructions; /// /// Gets of the default used by the agent. @@ -204,6 +204,8 @@ public override async IAsyncEnumerable RunStreamingAsync (ChatClientAgentThread safeThread, ChatOptions? chatOptions, List inputMessagesForChatClient, IList? aiContextProviderMessages) = await this.PrepareThreadAndMessagesAsync(thread, inputMessages, options, cancellationToken).ConfigureAwait(false); + ValidateStreamResumptionAllowed(chatOptions?.ContinuationToken, safeThread); + var chatClient = this.ChatClient; chatClient = ApplyRunOptionsTransformations(options, chatClient); @@ -270,7 +272,7 @@ public override async IAsyncEnumerable RunStreamingAsync this.UpdateThreadWithTypeAndConversationId(safeThread, chatResponse.ConversationId); // To avoid inconsistent state we only notify the thread of the input messages if no error occurs after the initial request. - await NotifyThreadOfNewMessagesAsync(safeThread, inputMessages.Concat(aiContextProviderMessages ?? []).Concat(chatResponse.Messages), cancellationToken).ConfigureAwait(false); + await NotifyMessageStoreOfNewMessagesAsync(safeThread, inputMessages.Concat(aiContextProviderMessages ?? []).Concat(chatResponse.Messages), cancellationToken).ConfigureAwait(false); // Notify the AIContextProvider of all new messages. await NotifyAIContextProviderOfSuccessAsync(safeThread, inputMessages, aiContextProviderMessages, chatResponse.Messages, cancellationToken).ConfigureAwait(false); @@ -281,12 +283,15 @@ public override async IAsyncEnumerable RunStreamingAsync base.GetService(serviceType, serviceKey) ?? (serviceType == typeof(AIAgentMetadata) ? this._agentMetadata : serviceType == typeof(IChatClient) ? this.ChatClient + : serviceType == typeof(ChatOptions) ? this._agentOptions?.ChatOptions + : serviceType == typeof(ChatClientAgentOptions) ? this._agentOptions : this.ChatClient.GetService(serviceType, serviceKey)); /// public override AgentThread GetNewThread() => new ChatClientAgentThread { + MessageStore = this._agentOptions?.ChatMessageStoreFactory?.Invoke(new() { SerializedState = default, JsonSerializerOptions = null }), AIContextProvider = this._agentOptions?.AIContextProviderFactory?.Invoke(new() { SerializedState = default, JsonSerializerOptions = null }) }; @@ -314,6 +319,34 @@ public AgentThread GetNewThread(string conversationId) AIContextProvider = this._agentOptions?.AIContextProviderFactory?.Invoke(new() { SerializedState = default, JsonSerializerOptions = null }) }; + /// + /// Creates a new agent thread instance using an existing to continue a conversation. + /// + /// The instance to use for managing the conversation's message history. + /// + /// A new instance configured to work with the provided . + /// + /// + /// + /// This method creates threads that do not support server-side conversation storage. + /// Some AI services require server-side conversation storage to function properly, and creating a thread + /// with a may not be compatible with these services. + /// + /// + /// Where a service requires server-side conversation storage, use . + /// + /// + /// If the agent detects, during the first run, that the underlying AI service requires server-side conversation storage, + /// the thread will throw an exception to indicate that it cannot continue using the provided . + /// + /// + public AgentThread GetNewThread(ChatMessageStore chatMessageStore) + => new ChatClientAgentThread() + { + MessageStore = Throw.IfNull(chatMessageStore), + AIContextProvider = this._agentOptions?.AIContextProviderFactory?.Invoke(new() { SerializedState = default, JsonSerializerOptions = null }) + }; + /// public override AgentThread DeserializeThread(JsonElement serializedThread, JsonSerializerOptions? jsonSerializerOptions = null) { @@ -382,7 +415,7 @@ private async Task RunCoreAsync inputMessagesForChatClient = []; IList? aiContextProviderMessages = null; @@ -646,12 +691,6 @@ await thread.AIContextProvider.InvokedAsync(new(inputMessages, aiContextProvider """); } - if (!string.IsNullOrWhiteSpace(this.Instructions)) - { - chatOptions ??= new(); - chatOptions.Instructions = string.IsNullOrWhiteSpace(chatOptions.Instructions) ? this.Instructions : $"{this.Instructions}\n{chatOptions.Instructions}"; - } - // Only create or update ChatOptions if we have an id on the thread and we don't have the same one already in ChatOptions. if (!string.IsNullOrWhiteSpace(typedThread.ConversationId) && typedThread.ConversationId != chatOptions?.ConversationId) { @@ -680,9 +719,45 @@ private void UpdateThreadWithTypeAndConversationId(ChatClientAgentThread thread, else { // If the service doesn't use service side thread storage (i.e. we got no id back from invocation), and - // the thread has no MessageStore yet, and we have a custom messages store, we should update the thread - // with the custom MessageStore so that it has somewhere to store the chat history. - thread.MessageStore ??= this._agentOptions?.ChatMessageStoreFactory?.Invoke(new() { SerializedState = default, JsonSerializerOptions = null }); + // the thread has no MessageStore yet, we should update the thread with the custom MessageStore or + // default InMemoryMessageStore so that it has somewhere to store the chat history. + thread.MessageStore ??= this._agentOptions?.ChatMessageStoreFactory?.Invoke(new() { SerializedState = default, JsonSerializerOptions = null }) ?? new InMemoryChatMessageStore(); + } + } + + private static Task NotifyMessageStoreOfNewMessagesAsync(ChatClientAgentThread thread, IEnumerable newMessages, CancellationToken cancellationToken) + { + var messageStore = thread.MessageStore; + + // Only notify the message store if we have one. + // If we don't have one, it means that the chat history is service managed and the underlying service is responsible for storing messages. + if (messageStore is not null) + { + return messageStore.AddMessagesAsync(newMessages, cancellationToken); + } + + return Task.CompletedTask; + } + + private static void ValidateStreamResumptionAllowed(ResponseContinuationToken? continuationToken, ChatClientAgentThread safeThread) + { + if (continuationToken is null) + { + return; + } + + // Streaming resumption is only supported with chat history managed by the agent service because, currently, there's no good solution + // to collect updates received in failed runs and pass them to the last successful run so it can store them to the message store. + if (safeThread.ConversationId is null) + { + throw new NotSupportedException("Streaming resumption is only supported when chat history is stored and managed by the underlying AI service."); + } + + // Similarly, streaming resumption is not supported when a context provider is used because, currently, there's no good solution + // to collect updates received in failed runs and pass them to the last successful run so it can notify the context provider of the updates. + if (safeThread.AIContextProvider is not null) + { + throw new NotSupportedException("Using context provider with streaming resumption is not supported."); } } diff --git a/dotnet/src/Microsoft.Agents.AI/ChatClient/ChatClientAgentOptions.cs b/dotnet/src/Microsoft.Agents.AI/ChatClient/ChatClientAgentOptions.cs index f83e6912d5..4a72d66f2d 100644 --- a/dotnet/src/Microsoft.Agents.AI/ChatClient/ChatClientAgentOptions.cs +++ b/dotnet/src/Microsoft.Agents.AI/ChatClient/ChatClientAgentOptions.cs @@ -1,7 +1,6 @@ // Copyright (c) Microsoft. All rights reserved. using System; -using System.Collections.Generic; using System.Text.Json; using Microsoft.Extensions.AI; @@ -15,37 +14,8 @@ namespace Microsoft.Agents.AI; /// identifier, display name, operational instructions, and a descriptive summary. It can be used to store and transfer /// agent-related metadata within a chat application. /// -public class ChatClientAgentOptions +public sealed class ChatClientAgentOptions { - /// - /// Initializes a new instance of the class. - /// - public ChatClientAgentOptions() - { - } - - /// - /// Initializes a new instance of the class with the specified parameters. - /// - /// If is provided, a new instance is created - /// with the specified instructions and tools. - /// The instructions or guidelines for the chat client agent. Can be if not specified. - /// The name of the chat client agent. Can be if not specified. - /// The description of the chat client agent. Can be if not specified. - /// A list of instances available to the chat client agent. Can be if no - /// tools are specified. - public ChatClientAgentOptions(string? instructions, string? name = null, string? description = null, IList? tools = null) - { - this.Name = name; - this.Instructions = instructions; - this.Description = description; - - if (tools is not null) - { - (this.ChatOptions ??= new()).Tools = tools; - } - } - /// /// Gets or sets the agent id. /// @@ -56,11 +26,6 @@ public ChatClientAgentOptions(string? instructions, string? name = null, string? /// public string? Name { get; set; } - /// - /// Gets or sets the agent instructions. - /// - public string? Instructions { get; set; } - /// /// Gets or sets the agent description. /// @@ -106,7 +71,6 @@ public ChatClientAgentOptions Clone() { Id = this.Id, Name = this.Name, - Instructions = this.Instructions, Description = this.Description, ChatOptions = this.ChatOptions?.Clone(), ChatMessageStoreFactory = this.ChatMessageStoreFactory, diff --git a/dotnet/src/Microsoft.Agents.AI/ChatClient/ChatClientAgentThread.cs b/dotnet/src/Microsoft.Agents.AI/ChatClient/ChatClientAgentThread.cs index baa36c0054..7f0ce9a1ea 100644 --- a/dotnet/src/Microsoft.Agents.AI/ChatClient/ChatClientAgentThread.cs +++ b/dotnet/src/Microsoft.Agents.AI/ChatClient/ChatClientAgentThread.cs @@ -1,12 +1,8 @@ // Copyright (c) Microsoft. All rights reserved. using System; -using System.Collections.Generic; using System.Diagnostics; using System.Text.Json; -using System.Threading; -using System.Threading.Tasks; -using Microsoft.Extensions.AI; using Microsoft.Shared.Diagnostics; namespace Microsoft.Agents.AI; @@ -17,7 +13,6 @@ namespace Microsoft.Agents.AI; [DebuggerDisplay("{DebuggerDisplay,nq}")] public class ChatClientAgentThread : AgentThread { - private string? _conversationId; private ChatMessageStore? _messageStore; /// @@ -75,8 +70,8 @@ internal ChatClientAgentThread( /// /// /// Note that either or may be set, but not both. - /// If is not null, and is set, - /// will be reverted to null, and vice versa. + /// If is not null, setting will throw an + /// exception. /// /// /// This property may be null in the following cases: @@ -91,12 +86,13 @@ internal ChatClientAgentThread( /// to fork the thread with each iteration. /// /// + /// Attempted to set a conversation ID but a is already set. public string? ConversationId { - get => this._conversationId; + get; internal set { - if (string.IsNullOrWhiteSpace(this._conversationId) && string.IsNullOrWhiteSpace(value)) + if (string.IsNullOrWhiteSpace(field) && string.IsNullOrWhiteSpace(value)) { return; } @@ -109,7 +105,7 @@ internal set throw new InvalidOperationException("Only the ConversationId or MessageStore may be set, but not both and switching from one to another is not supported."); } - this._conversationId = Throw.IfNullOrWhitespace(value); + field = Throw.IfNullOrWhitespace(value); } } @@ -140,7 +136,7 @@ internal set return; } - if (!string.IsNullOrWhiteSpace(this._conversationId)) + if (!string.IsNullOrWhiteSpace(this.ConversationId)) { // If we have a conversation id already, we shouldn't switch the thread to use a message store // since it means that the thread will not work with the original agent anymore. @@ -166,8 +162,8 @@ public override JsonElement Serialize(JsonSerializerOptions? jsonSerializerOptio var state = new ThreadState { ConversationId = this.ConversationId, - StoreState = storeState, - AIContextProviderState = aiContextProviderState + StoreState = storeState is { ValueKind: not JsonValueKind.Undefined } ? storeState : null, + AIContextProviderState = aiContextProviderState is { ValueKind: not JsonValueKind.Undefined } ? aiContextProviderState : null, }; return JsonSerializer.SerializeToElement(state, AgentJsonUtilities.DefaultOptions.GetTypeInfo(typeof(ThreadState))); @@ -181,36 +177,9 @@ public override JsonElement Serialize(JsonSerializerOptions? jsonSerializerOptio ?? this.AIContextProvider?.GetService(serviceType, serviceKey) ?? this.MessageStore?.GetService(serviceType, serviceKey); - /// - protected override async Task MessagesReceivedAsync(IEnumerable newMessages, CancellationToken cancellationToken = default) - { - switch (this) - { - case { ConversationId: not null }: - // If the thread messages are stored in the service - // there is nothing to do here, since invoking the - // service should already update the thread. - break; - - case { MessageStore: null }: - // If there is no conversation id, and no store we can createa a default in memory store and add messages to it. - this._messageStore = new InMemoryChatMessageStore(); - await this._messageStore!.AddMessagesAsync(newMessages, cancellationToken).ConfigureAwait(false); - break; - - case { MessageStore: not null }: - // If a store has been provided, we need to add the messages to the store. - await this._messageStore!.AddMessagesAsync(newMessages, cancellationToken).ConfigureAwait(false); - break; - - default: - throw new UnreachableException(); - } - } - [DebuggerBrowsable(DebuggerBrowsableState.Never)] private string DebuggerDisplay => - this._conversationId is { } conversationId ? $"ConversationId = {conversationId}" : + this.ConversationId is { } conversationId ? $"ConversationId = {conversationId}" : this._messageStore is InMemoryChatMessageStore inMemoryStore ? $"Count = {inMemoryStore.Count}" : this._messageStore is { } store ? $"Store = {store.GetType().Name}" : "Count = 0"; diff --git a/dotnet/src/Microsoft.Agents.AI/FunctionInvocationDelegatingAgentBuilderExtensions.cs b/dotnet/src/Microsoft.Agents.AI/FunctionInvocationDelegatingAgentBuilderExtensions.cs new file mode 100644 index 0000000000..5ff23f600c --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI/FunctionInvocationDelegatingAgentBuilderExtensions.cs @@ -0,0 +1,52 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Threading; +using System.Threading.Tasks; +using Microsoft.Extensions.AI; +using Microsoft.Shared.Diagnostics; + +namespace Microsoft.Agents.AI; + +/// +/// Provides extension methods for configuring and customizing instances. +/// +public static class FunctionInvocationDelegatingAgentBuilderExtensions +{ + /// + /// Adds function invocation callbacks to the pipeline that intercepts and processes calls. + /// + /// The to which the function invocation callback is added. + /// + /// A delegate that processes function invocations. The delegate receives the instance, + /// the function invocation context, and a continuation delegate representing the next callback in the pipeline. + /// It returns a task representing the result of the function invocation. + /// + /// The instance with the function invocation callback added, enabling method chaining. + /// or is . + /// + /// + /// The callback must call the provided continuation delegate to proceed with the function invocation, + /// unless it intends to completely replace the function's behavior. + /// + /// + /// The inner agent or the pipeline wrapping it must include a . If one does not exist, + /// the added to the pipline by this method will throw an exception when it is invoked. + /// + /// + public static AIAgentBuilder Use(this AIAgentBuilder builder, Func>, CancellationToken, ValueTask> callback) + { + _ = Throw.IfNull(builder); + _ = Throw.IfNull(callback); + return builder.Use((innerAgent, _) => + { + // Function calling requires a ChatClientAgent inner agent. + if (innerAgent.GetService() is null) + { + throw new InvalidOperationException($"The function invocation middleware can only be used with decorations of a {nameof(AIAgent)} that support usage of FunctionInvokingChatClient decorated chat clients."); + } + + return new FunctionInvocationDelegatingAgent(innerAgent, callback); + }); + } +} diff --git a/dotnet/src/Microsoft.Agents.AI/LoggingAgent.cs b/dotnet/src/Microsoft.Agents.AI/LoggingAgent.cs new file mode 100644 index 0000000000..b986e58bae --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI/LoggingAgent.cs @@ -0,0 +1,209 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Collections.Generic; +using System.Runtime.CompilerServices; +using System.Text.Json; +using System.Threading; +using System.Threading.Tasks; +using Microsoft.Extensions.AI; +using Microsoft.Extensions.Logging; +using Microsoft.Shared.Diagnostics; +using LogLevel = Microsoft.Extensions.Logging.LogLevel; + +namespace Microsoft.Agents.AI; + +/// +/// A delegating AI agent that logs agent operations to an . +/// +/// +/// +/// The provided implementation of is thread-safe for concurrent use so long as the +/// employed is also thread-safe for concurrent use. +/// +/// +/// When the employed enables , the contents of +/// messages, options, and responses are logged. These may contain sensitive application data. +/// is disabled by default and should never be enabled in a production environment. +/// Messages and options are not logged at other logging levels. +/// +/// +public sealed partial class LoggingAgent : DelegatingAIAgent +{ + /// An instance used for all logging. + private readonly ILogger _logger; + + /// The to use for serialization of state written to the logger. + private JsonSerializerOptions _jsonSerializerOptions; + + /// Initializes a new instance of the class. + /// The underlying . + /// An instance that will be used for all logging. + /// or is . + public LoggingAgent(AIAgent innerAgent, ILogger logger) + : base(innerAgent) + { + this._logger = Throw.IfNull(logger); + this._jsonSerializerOptions = AgentJsonUtilities.DefaultOptions; + } + + /// Gets or sets JSON serialization options to use when serializing logging data. + public JsonSerializerOptions JsonSerializerOptions + { + get => this._jsonSerializerOptions; + set => this._jsonSerializerOptions = Throw.IfNull(value); + } + + /// + public override async Task RunAsync( + IEnumerable messages, AgentThread? thread = null, AgentRunOptions? options = null, CancellationToken cancellationToken = default) + { + if (this._logger.IsEnabled(LogLevel.Debug)) + { + if (this._logger.IsEnabled(LogLevel.Trace)) + { + this.LogInvokedSensitive(nameof(RunAsync), this.AsJson(messages), this.AsJson(options), this.AsJson(this.GetService())); + } + else + { + this.LogInvoked(nameof(RunAsync)); + } + } + + try + { + AgentRunResponse response = await base.RunAsync(messages, thread, options, cancellationToken).ConfigureAwait(false); + + if (this._logger.IsEnabled(LogLevel.Debug)) + { + if (this._logger.IsEnabled(LogLevel.Trace)) + { + this.LogCompletedSensitive(nameof(RunAsync), this.AsJson(response)); + } + else + { + this.LogCompleted(nameof(RunAsync)); + } + } + + return response; + } + catch (OperationCanceledException) + { + this.LogInvocationCanceled(nameof(RunAsync)); + throw; + } + catch (Exception ex) + { + this.LogInvocationFailed(nameof(RunAsync), ex); + throw; + } + } + + /// + public override async IAsyncEnumerable RunStreamingAsync( + IEnumerable messages, AgentThread? thread = null, AgentRunOptions? options = null, [EnumeratorCancellation] CancellationToken cancellationToken = default) + { + if (this._logger.IsEnabled(LogLevel.Debug)) + { + if (this._logger.IsEnabled(LogLevel.Trace)) + { + this.LogInvokedSensitive(nameof(RunStreamingAsync), this.AsJson(messages), this.AsJson(options), this.AsJson(this.GetService())); + } + else + { + this.LogInvoked(nameof(RunStreamingAsync)); + } + } + + IAsyncEnumerator e; + try + { + e = base.RunStreamingAsync(messages, thread, options, cancellationToken).GetAsyncEnumerator(cancellationToken); + } + catch (OperationCanceledException) + { + this.LogInvocationCanceled(nameof(RunStreamingAsync)); + throw; + } + catch (Exception ex) + { + this.LogInvocationFailed(nameof(RunStreamingAsync), ex); + throw; + } + + try + { + AgentRunResponseUpdate? update = null; + while (true) + { + try + { + if (!await e.MoveNextAsync().ConfigureAwait(false)) + { + break; + } + + update = e.Current; + } + catch (OperationCanceledException) + { + this.LogInvocationCanceled(nameof(RunStreamingAsync)); + throw; + } + catch (Exception ex) + { + this.LogInvocationFailed(nameof(RunStreamingAsync), ex); + throw; + } + + if (this._logger.IsEnabled(LogLevel.Trace)) + { + this.LogStreamingUpdateSensitive(this.AsJson(update)); + } + + yield return update; + } + + this.LogCompleted(nameof(RunStreamingAsync)); + } + finally + { + await e.DisposeAsync().ConfigureAwait(false); + } + } + + private string AsJson(T value) + { + try + { + return JsonSerializer.Serialize(value, this._jsonSerializerOptions.GetTypeInfo(typeof(T))); + } + catch + { + // If serialization fails, return a simple string representation + return value?.ToString() ?? "null"; + } + } + + [LoggerMessage(LogLevel.Debug, "{MethodName} invoked.")] + private partial void LogInvoked(string methodName); + + [LoggerMessage(LogLevel.Trace, "{MethodName} invoked: {Messages}. Options: {Options}. Metadata: {Metadata}.")] + private partial void LogInvokedSensitive(string methodName, string messages, string options, string metadata); + + [LoggerMessage(LogLevel.Debug, "{MethodName} completed.")] + private partial void LogCompleted(string methodName); + + [LoggerMessage(LogLevel.Trace, "{MethodName} completed: {Response}.")] + private partial void LogCompletedSensitive(string methodName, string response); + + [LoggerMessage(LogLevel.Trace, "RunStreamingAsync received update: {Update}")] + private partial void LogStreamingUpdateSensitive(string update); + + [LoggerMessage(LogLevel.Debug, "{MethodName} canceled.")] + private partial void LogInvocationCanceled(string methodName); + + [LoggerMessage(LogLevel.Error, "{MethodName} failed.")] + private partial void LogInvocationFailed(string methodName, Exception error); +} diff --git a/dotnet/src/Microsoft.Agents.AI/LoggingAgentBuilderExtensions.cs b/dotnet/src/Microsoft.Agents.AI/LoggingAgentBuilderExtensions.cs new file mode 100644 index 0000000000..c4de608364 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI/LoggingAgentBuilderExtensions.cs @@ -0,0 +1,66 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using Microsoft.Extensions.DependencyInjection; +using Microsoft.Extensions.Logging; +using Microsoft.Extensions.Logging.Abstractions; +using Microsoft.Shared.Diagnostics; +using LogLevel = Microsoft.Extensions.Logging.LogLevel; + +namespace Microsoft.Agents.AI; + +/// +/// Provides extension methods for adding logging support to instances. +/// +public static class LoggingAgentBuilderExtensions +{ + /// + /// Adds logging to the agent pipeline, enabling detailed observability of agent operations. + /// + /// The to which logging support will be added. + /// + /// An optional used to create a logger with which logging should be performed. + /// If not supplied, a required instance will be resolved from the service provider. + /// + /// + /// An optional callback that provides additional configuration of the instance. + /// This allows for fine-tuning logging behavior such as customizing JSON serialization options. + /// + /// The with logging support added, enabling method chaining. + /// is . + /// + /// + /// When the employed enables , the contents of + /// messages, options, and responses are logged. These may contain sensitive application data. + /// is disabled by default and should never be enabled in a production environment. + /// Messages and options are not logged at other logging levels. + /// + /// + /// If the resolved or provided is , this will be a no-op where + /// logging will be effectively disabled. In this case, the will not be added. + /// + /// + public static AIAgentBuilder UseLogging( + this AIAgentBuilder builder, + ILoggerFactory? loggerFactory = null, + Action? configure = null) + { + _ = Throw.IfNull(builder); + + return builder.Use((innerAgent, services) => + { + loggerFactory ??= services.GetRequiredService(); + + // If the factory we resolve is for the null logger, the LoggingAgent will end up + // being an expensive nop, so skip adding it and just return the inner agent. + if (loggerFactory == NullLoggerFactory.Instance) + { + return innerAgent; + } + + LoggingAgent agent = new(innerAgent, loggerFactory.CreateLogger(nameof(LoggingAgent))); + configure?.Invoke(agent); + return agent; + }); + } +} diff --git a/dotnet/src/Microsoft.Agents.AI/Memory/ChatHistoryMemoryProvider.cs b/dotnet/src/Microsoft.Agents.AI/Memory/ChatHistoryMemoryProvider.cs new file mode 100644 index 0000000000..6384d36d9f --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI/Memory/ChatHistoryMemoryProvider.cs @@ -0,0 +1,501 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Collections.Generic; +using System.Linq; +using System.Linq.Expressions; +using System.Text.Json; +using System.Threading; +using System.Threading.Tasks; +using Microsoft.Extensions.AI; +using Microsoft.Extensions.Logging; +using Microsoft.Extensions.VectorData; +using Microsoft.Shared.Diagnostics; + +namespace Microsoft.Agents.AI; + +/// +/// A context provider that stores all chat history in a vector store and is able to +/// retrieve related chat history later to augment the current conversation. +/// +/// +/// +/// This provider stores chat messages in a vector store and retrieves relevant previous messages +/// to provide as context during agent invocations. It uses the VectorStore and VectorStoreCollection +/// abstractions to work with any compatible vector store implementation. +/// +/// +/// Messages are stored during the method and retrieved during the +/// method using semantic similarity search. +/// +/// +/// Behavior is configurable through . When +/// is selected the provider +/// exposes a function tool that the model can invoke to retrieve relevant memories on demand instead of +/// injecting them automatically on each invocation. +/// +/// +public sealed class ChatHistoryMemoryProvider : AIContextProvider, IDisposable +{ + private const string DefaultContextPrompt = "## Memories\nConsider the following memories when answering user questions:"; + private const int DefaultMaxResults = 3; + private const string DefaultFunctionToolName = "Search"; + private const string DefaultFunctionToolDescription = "Allows searching for related previous chat history to help answer the user question."; + + private readonly VectorStore _vectorStore; + private readonly VectorStoreCollection> _collection; + private readonly int _maxResults; + private readonly string _contextPrompt; + private readonly bool _enableSensitiveTelemetryData; + private readonly ChatHistoryMemoryProviderOptions.SearchBehavior _searchTime; + private readonly AITool[] _tools; + private readonly ILogger? _logger; + + private readonly ChatHistoryMemoryProviderScope _storageScope; + private readonly ChatHistoryMemoryProviderScope _searchScope; + + private bool _collectionInitialized; + private readonly SemaphoreSlim _initializationLock = new(1, 1); + private bool _disposedValue; + + /// + /// Initializes a new instance of the class. + /// + /// The vector store to use for storing and retrieving chat history. + /// The name of the collection for storing chat history in the vector store. + /// The number of dimensions to use for the chat history vector store embeddings. + /// Optional values to scope the chat history storage with. + /// Optional values to scope the chat history search with. Where values are null, no filtering is done using those values. Defaults to if not provided. + /// Optional configuration options. + /// Optional logger factory. + /// Thrown when is . + public ChatHistoryMemoryProvider( + VectorStore vectorStore, + string collectionName, + int vectorDimensions, + ChatHistoryMemoryProviderScope storageScope, + ChatHistoryMemoryProviderScope? searchScope = null, + ChatHistoryMemoryProviderOptions? options = null, + ILoggerFactory? loggerFactory = null) + : this( + vectorStore, + collectionName, + vectorDimensions, + new ChatHistoryMemoryProviderState + { + StorageScope = new(Throw.IfNull(storageScope)), + SearchScope = searchScope ?? new(storageScope), + }, + options, + loggerFactory) + { + } + + /// + /// Initializes a new instance of the class from previously serialized state. + /// + /// The vector store to use for storing and retrieving chat history. + /// The name of the collection for storing chat history in the vector store. + /// The number of dimensions to use for the chat history vector store embeddings. + /// A representing the serialized state of the provider. + /// Optional settings for customizing the JSON deserialization process. + /// Optional configuration options. + /// Optional logger factory. + public ChatHistoryMemoryProvider( + VectorStore vectorStore, + string collectionName, + int vectorDimensions, + JsonElement serializedState, + JsonSerializerOptions? jsonSerializerOptions = null, + ChatHistoryMemoryProviderOptions? options = null, + ILoggerFactory? loggerFactory = null) + : this( + vectorStore, + collectionName, + vectorDimensions, + DeserializeState(serializedState, jsonSerializerOptions), + options, + loggerFactory) + { + } + + private ChatHistoryMemoryProvider( + VectorStore vectorStore, + string collectionName, + int vectorDimensions, + ChatHistoryMemoryProviderState? state = null, + ChatHistoryMemoryProviderOptions? options = null, + ILoggerFactory? loggerFactory = null) + { + this._vectorStore = vectorStore ?? throw new ArgumentNullException(nameof(vectorStore)); + options ??= new ChatHistoryMemoryProviderOptions(); + this._maxResults = options.MaxResults.HasValue ? Throw.IfLessThanOrEqual(options.MaxResults.Value, 0) : DefaultMaxResults; + this._contextPrompt = options.ContextPrompt ?? DefaultContextPrompt; + this._enableSensitiveTelemetryData = options.EnableSensitiveTelemetryData; + this._searchTime = options.SearchTime; + this._logger = loggerFactory?.CreateLogger(); + + if (state == null || state.StorageScope == null || state.SearchScope == null) + { + throw new InvalidOperationException($"The {nameof(ChatHistoryMemoryProvider)} state did not contain the required scope properties."); + } + + this._storageScope = state.StorageScope; + this._searchScope = state.SearchScope; + + // Create on-demand search tool (only used when behavior is OnDemandFunctionCalling) + this._tools = + [ + AIFunctionFactory.Create( + (Func>)this.SearchTextAsync, + name: options.FunctionToolName ?? DefaultFunctionToolName, + description: options.FunctionToolDescription ?? DefaultFunctionToolDescription) + ]; + + // Create a definition so that we can use the dimensions provided at runtime. + var definition = new VectorStoreCollectionDefinition + { + Properties = + [ + new VectorStoreKeyProperty("Key", typeof(Guid)), + new VectorStoreDataProperty("Role", typeof(string)) { IsIndexed = true }, + new VectorStoreDataProperty("MessageId", typeof(string)) { IsIndexed = true }, + new VectorStoreDataProperty("AuthorName", typeof(string)), + new VectorStoreDataProperty("ApplicationId", typeof(string)) { IsIndexed = true }, + new VectorStoreDataProperty("AgentId", typeof(string)) { IsIndexed = true }, + new VectorStoreDataProperty("UserId", typeof(string)) { IsIndexed = true }, + new VectorStoreDataProperty("ThreadId", typeof(string)) { IsIndexed = true }, + new VectorStoreDataProperty("Content", typeof(string)) { IsFullTextIndexed = true }, + new VectorStoreDataProperty("CreatedAt", typeof(string)) { IsIndexed = true }, + new VectorStoreVectorProperty("ContentEmbedding", typeof(string), Throw.IfLessThan(vectorDimensions, 1)) + ] + }; + + this._collection = this._vectorStore.GetDynamicCollection(Throw.IfNullOrWhitespace(collectionName), definition); + } + + /// + public override async ValueTask InvokingAsync(InvokingContext context, CancellationToken cancellationToken = default) + { + _ = Throw.IfNull(context); + + if (this._searchTime == ChatHistoryMemoryProviderOptions.SearchBehavior.OnDemandFunctionCalling) + { + // Expose search tool for on-demand invocation by the model + return new AIContext { Tools = this._tools }; + } + + try + { + // Get the text from the current request messages + var requestText = string.Join("\n", context.RequestMessages + .Where(m => m != null && !string.IsNullOrWhiteSpace(m.Text)) + .Select(m => m.Text)); + + if (string.IsNullOrWhiteSpace(requestText)) + { + return new AIContext(); + } + + // Search for relevant chat history + var contextText = await this.SearchTextAsync(requestText, cancellationToken).ConfigureAwait(false); + + if (string.IsNullOrWhiteSpace(contextText)) + { + return new AIContext(); + } + + return new AIContext + { + Messages = [new ChatMessage(ChatRole.User, contextText)] + }; + } + catch (Exception ex) + { + if (this._logger?.IsEnabled(LogLevel.Error) is true) + { + this._logger.LogError( + ex, + "ChatHistoryMemoryProvider: Failed to search for chat history due to error. ApplicationId: '{ApplicationId}', AgentId: '{AgentId}', ThreadId: '{ThreadId}', UserId: '{UserId}'.", + this._searchScope.ApplicationId, + this._searchScope.AgentId, + this._searchScope.ThreadId, + this.SanitizeLogData(this._searchScope.UserId)); + } + + return new AIContext(); + } + } + + /// + public override async ValueTask InvokedAsync(InvokedContext context, CancellationToken cancellationToken = default) + { + _ = Throw.IfNull(context); + + // Only store if invocation was successful + if (context.InvokeException != null) + { + return; + } + + try + { + // Ensure the collection is initialized + var collection = await this.EnsureCollectionExistsAsync(cancellationToken).ConfigureAwait(false); + + List> itemsToStore = context.RequestMessages + .Concat(context.ResponseMessages ?? []) + .Select(message => new Dictionary + { + ["Key"] = Guid.NewGuid(), + ["Role"] = message.Role.ToString(), + ["MessageId"] = message.MessageId, + ["AuthorName"] = message.AuthorName, + ["ApplicationId"] = this._storageScope?.ApplicationId, + ["AgentId"] = this._storageScope?.AgentId, + ["UserId"] = this._storageScope?.UserId, + ["ThreadId"] = this._storageScope?.ThreadId, + ["Content"] = message.Text, + ["CreatedAt"] = message.CreatedAt?.ToString("O") ?? DateTimeOffset.UtcNow.ToString("O"), + ["ContentEmbedding"] = message.Text, + }) + .ToList(); + + if (itemsToStore.Count > 0) + { + await collection.UpsertAsync(itemsToStore, cancellationToken).ConfigureAwait(false); + } + } + catch (Exception ex) + { + if (this._logger?.IsEnabled(LogLevel.Error) is true) + { + this._logger.LogError( + ex, + "ChatHistoryMemoryProvider: Failed to add messages to chat history vector store due to error. ApplicationId: '{ApplicationId}', AgentId: '{AgentId}', ThreadId: '{ThreadId}', UserId: '{UserId}'.", + this._searchScope.ApplicationId, + this._searchScope.AgentId, + this._searchScope.ThreadId, + this.SanitizeLogData(this._searchScope.UserId)); + } + } + } + + /// + /// Function callable by the AI model (when enabled) to perform an ad-hoc chat history search. + /// + /// The query text. + /// Cancellation token. + /// Formatted search results (may be empty). + internal async Task SearchTextAsync(string userQuestion, CancellationToken cancellationToken = default) + { + if (string.IsNullOrWhiteSpace(userQuestion)) + { + return string.Empty; + } + + var results = await this.SearchChatHistoryAsync(userQuestion, this._maxResults, cancellationToken).ConfigureAwait(false); + if (!results.Any()) + { + return string.Empty; + } + + // Format the results as a single context message + var outputResultsText = string.Join("\n", results.Select(x => (string?)x["Content"]).Where(c => !string.IsNullOrWhiteSpace(c))); + if (string.IsNullOrWhiteSpace(outputResultsText)) + { + return string.Empty; + } + + var formatted = $"{this._contextPrompt}\n{outputResultsText}"; + + if (this._logger?.IsEnabled(LogLevel.Trace) is true) + { + this._logger.LogTrace( + "ChatHistoryMemoryProvider: Search Results\nInput:{Input}\nOutput:{MessageText}\n ApplicationId: '{ApplicationId}', AgentId: '{AgentId}', ThreadId: '{ThreadId}', UserId: '{UserId}'.", + this.SanitizeLogData(userQuestion), + this.SanitizeLogData(formatted), + this._searchScope.ApplicationId, + this._searchScope.AgentId, + this._searchScope.ThreadId, + this.SanitizeLogData(this._searchScope.UserId)); + } + + return formatted; + } + + /// + /// Searches for relevant chat history items based on the provided query text. + /// + /// The text to search for. + /// The maximum number of results to return. + /// The cancellation token. + /// A list of relevant chat history items. + private async Task>> SearchChatHistoryAsync( + string queryText, + int top, + CancellationToken cancellationToken = default) + { + if (string.IsNullOrWhiteSpace(queryText)) + { + return []; + } + + var collection = await this.EnsureCollectionExistsAsync(cancellationToken).ConfigureAwait(false); + + string? applicationId = this._searchScope.ApplicationId; + string? agentId = this._searchScope.AgentId; + string? userId = this._searchScope.UserId; + string? threadId = this._searchScope.ThreadId; + + Expression, bool>>? filter = null; + if (applicationId != null) + { + filter = x => (string?)x["ApplicationId"] == applicationId; + } + + if (agentId != null) + { + Expression, bool>> agentIdFilter = x => (string?)x["AgentId"] == agentId; + filter = filter == null ? agentIdFilter : Expression.Lambda, bool>>( + Expression.AndAlso(filter.Body, agentIdFilter.Body), + filter.Parameters); + } + + if (userId != null) + { + Expression, bool>> userIdFilter = x => (string?)x["UserId"] == userId; + filter = filter == null ? userIdFilter : Expression.Lambda, bool>>( + Expression.AndAlso(filter.Body, userIdFilter.Body), + filter.Parameters); + } + + if (threadId != null) + { + Expression, bool>> threadIdFilter = x => (string?)x["ThreadId"] == threadId; + filter = filter == null ? threadIdFilter : Expression.Lambda, bool>>( + Expression.AndAlso(filter.Body, threadIdFilter.Body), + filter.Parameters); + } + + // Use search to find relevant messages + var searchResults = collection.SearchAsync( + queryText, + top, + options: new() + { + Filter = filter + }, + cancellationToken: cancellationToken); + + var results = new List>(); + await foreach (var result in searchResults.WithCancellation(cancellationToken).ConfigureAwait(false)) + { + results.Add(result.Record); + } + + if (this._logger?.IsEnabled(LogLevel.Information) is true) + { + this._logger.LogInformation( + "ChatHistoryMemoryProvider: Retrieved {Count} search results. ApplicationId: '{ApplicationId}', AgentId: '{AgentId}', ThreadId: '{ThreadId}', UserId: '{UserId}'.", + results.Count, + this._searchScope.ApplicationId, + this._searchScope.AgentId, + this._searchScope.ThreadId, + this.SanitizeLogData(this._searchScope.UserId)); + } + + return results; + } + + /// + /// Ensures the collection exists in the vector store, creating it if necessary. + /// + /// The cancellation token. + /// The vector store collection. + private async Task>> EnsureCollectionExistsAsync( + CancellationToken cancellationToken = default) + { + if (this._collectionInitialized) + { + return this._collection; + } + + await this._initializationLock.WaitAsync(cancellationToken).ConfigureAwait(false); + try + { + if (this._collectionInitialized) + { + return this._collection; + } + + await this._collection.EnsureCollectionExistsAsync(cancellationToken).ConfigureAwait(false); + this._collectionInitialized = true; + + return this._collection; + } + finally + { + this._initializationLock.Release(); + } + } + + /// + private void Dispose(bool disposing) + { + if (!this._disposedValue) + { + if (disposing) + { + this._initializationLock.Dispose(); + this._collection?.Dispose(); + } + + this._disposedValue = true; + } + } + + /// + public void Dispose() + { + // Do not change this code. Put cleanup code in 'Dispose(bool disposing)' method + this.Dispose(disposing: true); + GC.SuppressFinalize(this); + } + + /// + /// Serializes the current provider state to a including storage and search scopes. + /// + /// Optional serializer options. + /// Serialized provider state. + public override JsonElement Serialize(JsonSerializerOptions? jsonSerializerOptions = null) + { + var state = new ChatHistoryMemoryProviderState + { + StorageScope = this._storageScope, + SearchScope = this._searchScope, + }; + + var jso = jsonSerializerOptions ?? AgentJsonUtilities.DefaultOptions; + return JsonSerializer.SerializeToElement(state, jso.GetTypeInfo(typeof(ChatHistoryMemoryProviderState))); + } + + private static ChatHistoryMemoryProviderState? DeserializeState(JsonElement serializedState, JsonSerializerOptions? jsonSerializerOptions) + { + if (serializedState.ValueKind != JsonValueKind.Object) + { + return null; + } + + var jso = jsonSerializerOptions ?? AgentJsonUtilities.DefaultOptions; + return serializedState.Deserialize(jso.GetTypeInfo(typeof(ChatHistoryMemoryProviderState))) as ChatHistoryMemoryProviderState; + } + + private string? SanitizeLogData(string? data) => this._enableSensitiveTelemetryData ? data : ""; + + internal sealed class ChatHistoryMemoryProviderState + { + public ChatHistoryMemoryProviderScope? StorageScope { get; set; } + public ChatHistoryMemoryProviderScope? SearchScope { get; set; } + } +} diff --git a/dotnet/src/Microsoft.Agents.AI/Memory/ChatHistoryMemoryProviderOptions.cs b/dotnet/src/Microsoft.Agents.AI/Memory/ChatHistoryMemoryProviderOptions.cs new file mode 100644 index 0000000000..e09de68a59 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI/Memory/ChatHistoryMemoryProviderOptions.cs @@ -0,0 +1,62 @@ +// Copyright (c) Microsoft. All rights reserved. + +namespace Microsoft.Agents.AI; + +/// +/// Options controlling the behavior of . +/// +public sealed class ChatHistoryMemoryProviderOptions +{ + /// + /// Gets or sets a value indicating when the search should be executed. + /// + /// by default. + public SearchBehavior SearchTime { get; set; } = SearchBehavior.BeforeAIInvoke; + + /// + /// Gets or sets the name of the exposed search tool when operating in on-demand mode. + /// + /// Defaults to "Search". + public string? FunctionToolName { get; set; } + + /// + /// Gets or sets the description of the exposed search tool when operating in on-demand mode. + /// + /// Defaults to "Allows searching through previous chat history to help answer the user question.". + public string? FunctionToolDescription { get; set; } + + /// + /// Gets or sets the context prompt prefixed to results. + /// + public string? ContextPrompt { get; set; } + + /// + /// Gets or sets the maximum number of results to retrieve from the chat history. + /// + /// + /// Defaults to 3 if not set. + /// + public int? MaxResults { get; set; } + + /// + /// Gets or sets a value indicating whether sensitive data such as user ids and user messages may appear in logs. + /// + /// Defaults to . + public bool EnableSensitiveTelemetryData { get; set; } + + /// + /// Behavior choices for the provider. + /// + public enum SearchBehavior + { + /// + /// Execute search prior to each invocation and inject results as a message. + /// + BeforeAIInvoke, + + /// + /// Expose a function tool to perform search on-demand via function/tool calling. + /// + OnDemandFunctionCalling + } +} diff --git a/dotnet/src/Microsoft.Agents.AI/Memory/ChatHistoryMemoryProviderScope.cs b/dotnet/src/Microsoft.Agents.AI/Memory/ChatHistoryMemoryProviderScope.cs new file mode 100644 index 0000000000..2715ed2e20 --- /dev/null +++ b/dotnet/src/Microsoft.Agents.AI/Memory/ChatHistoryMemoryProviderScope.cs @@ -0,0 +1,53 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Microsoft.Shared.Diagnostics; + +namespace Microsoft.Agents.AI; + +/// +/// Allows scoping of chat history for the . +/// +public sealed class ChatHistoryMemoryProviderScope +{ + /// + /// Initializes a new instance of the class. + /// + public ChatHistoryMemoryProviderScope() { } + + /// + /// Initializes a new instance of the class by cloning an existing scope. + /// + /// The scope to clone. + public ChatHistoryMemoryProviderScope(ChatHistoryMemoryProviderScope sourceScope) + { + Throw.IfNull(sourceScope); + + this.ApplicationId = sourceScope.ApplicationId; + this.AgentId = sourceScope.AgentId; + this.ThreadId = sourceScope.ThreadId; + this.UserId = sourceScope.UserId; + } + + /// + /// Gets or sets an optional ID for the application to scope chat history to. + /// + /// If not set, the scope of the chat history will span all applications. + public string? ApplicationId { get; set; } + + /// + /// Gets or sets an optional ID for the agent to scope chat history to. + /// + /// If not set, the scope of the chat history will span all agents. + public string? AgentId { get; set; } + + /// + /// Gets or sets an optional ID for the thread to scope chat history to. + /// + public string? ThreadId { get; set; } + + /// + /// Gets or sets an optional ID for the user to scope chat history to. + /// + /// If not set, the scope of the chat history will span all users. + public string? UserId { get; set; } +} diff --git a/dotnet/src/Microsoft.Agents.AI/Microsoft.Agents.AI.csproj b/dotnet/src/Microsoft.Agents.AI/Microsoft.Agents.AI.csproj index 59345d21ae..0a9eec9763 100644 --- a/dotnet/src/Microsoft.Agents.AI/Microsoft.Agents.AI.csproj +++ b/dotnet/src/Microsoft.Agents.AI/Microsoft.Agents.AI.csproj @@ -1,8 +1,6 @@  - $(ProjectsTargetFrameworks) - $(ProjectsDebugTargetFrameworks) preview $(NoWarn);MEAI001 @@ -21,6 +19,7 @@ + @@ -31,8 +30,11 @@ - + + + + diff --git a/dotnet/src/Microsoft.Agents.AI/AIAgentBuilderExtensions.cs b/dotnet/src/Microsoft.Agents.AI/OpenTelemetryAgentBuilderExtensions.cs similarity index 51% rename from dotnet/src/Microsoft.Agents.AI/AIAgentBuilderExtensions.cs rename to dotnet/src/Microsoft.Agents.AI/OpenTelemetryAgentBuilderExtensions.cs index a5a280d039..8f83a8dda1 100644 --- a/dotnet/src/Microsoft.Agents.AI/AIAgentBuilderExtensions.cs +++ b/dotnet/src/Microsoft.Agents.AI/OpenTelemetryAgentBuilderExtensions.cs @@ -1,55 +1,15 @@ // Copyright (c) Microsoft. All rights reserved. using System; -using System.Threading; -using System.Threading.Tasks; -using Microsoft.Extensions.AI; using Microsoft.Shared.Diagnostics; namespace Microsoft.Agents.AI; /// -/// Provides extension methods for configuring and customizing instances. +/// Provides extension methods for adding OpenTelemetry instrumentation to instances. /// -public static class AIAgentBuilderExtensions +public static class OpenTelemetryAgentBuilderExtensions { - /// - /// Adds function invocation callbacks to the pipeline that intercepts and processes calls. - /// - /// The to which the function invocation callback is added. - /// - /// A delegate that processes function invocations. The delegate receives the instance, - /// the function invocation context, and a continuation delegate representing the next callback in the pipeline. - /// It returns a task representing the result of the function invocation. - /// - /// The instance with the function invocation callback added, enabling method chaining. - /// or is . - /// - /// - /// The callback must call the provided continuation delegate to proceed with the function invocation, - /// unless it intends to completely replace the function's behavior. - /// - /// - /// The inner agent or the pipeline wrapping it must include a . If one does not exist, - /// the added to the pipline by this method will throw an exception when it is invoked. - /// - /// - public static AIAgentBuilder Use(this AIAgentBuilder builder, Func>, CancellationToken, ValueTask> callback) - { - _ = Throw.IfNull(builder); - _ = Throw.IfNull(callback); - return builder.Use((innerAgent, _) => - { - // Function calling requires a ChatClientAgent inner agent. - if (innerAgent.GetService() is null) - { - throw new InvalidOperationException($"The function invocation middleware can only be used with decorations of a {nameof(AIAgent)} that support usage of FunctionInvokingChatClient decorated chat clients."); - } - - return new FunctionInvocationDelegatingAgent(innerAgent, callback); - }); - } - /// /// Adds OpenTelemetry instrumentation to the agent pipeline, enabling comprehensive observability for agent operations. /// diff --git a/dotnet/src/Microsoft.Agents.AI/Data/TextSearchProvider.cs b/dotnet/src/Microsoft.Agents.AI/TextSearchProvider.cs similarity index 93% rename from dotnet/src/Microsoft.Agents.AI/Data/TextSearchProvider.cs rename to dotnet/src/Microsoft.Agents.AI/TextSearchProvider.cs index f76629a577..be9eba1365 100644 --- a/dotnet/src/Microsoft.Agents.AI/Data/TextSearchProvider.cs +++ b/dotnet/src/Microsoft.Agents.AI/TextSearchProvider.cs @@ -11,7 +11,7 @@ using Microsoft.Extensions.Logging; using Microsoft.Shared.Diagnostics; -namespace Microsoft.Agents.AI.Data; +namespace Microsoft.Agents.AI; /// /// A text search context provider that performs a search over external knowledge @@ -134,7 +134,11 @@ public override async ValueTask InvokingAsync(InvokingContext context // Search var results = await this._searchAsync(input, cancellationToken).ConfigureAwait(false); IList materialized = results as IList ?? results.ToList(); - this._logger?.LogInformation("TextSearchProvider: Retrieved {Count} search results.", materialized.Count); + + if (this._logger?.IsEnabled(LogLevel.Information) is true) + { + this._logger?.LogInformation("TextSearchProvider: Retrieved {Count} search results.", materialized.Count); + } if (materialized.Count == 0) { @@ -144,7 +148,10 @@ public override async ValueTask InvokingAsync(InvokingContext context // Format search results string formatted = this.FormatResults(materialized); - this._logger?.LogTrace("TextSearchProvider: Search Results\nInput:{Input}\nOutput:{MessageText}", input, formatted); + if (this._logger?.IsEnabled(LogLevel.Trace) is true) + { + this._logger.LogTrace("TextSearchProvider: Search Results\nInput:{Input}\nOutput:{MessageText}", input, formatted); + } return new AIContext { @@ -230,8 +237,15 @@ internal async Task SearchAsync(string userQuestion, CancellationToken c IList materialized = results as IList ?? results.ToList(); string outputText = this.FormatResults(materialized); - this._logger?.LogInformation("TextSearchProvider: Retrieved {Count} search results.", materialized.Count); - this._logger?.LogTrace("TextSearchProvider Input:{UserQuestion}\nOutput:{MessageText}", userQuestion, outputText); + if (this._logger?.IsEnabled(LogLevel.Information) is true) + { + this._logger.LogInformation("TextSearchProvider: Retrieved {Count} search results.", materialized.Count); + + if (this._logger.IsEnabled(LogLevel.Trace)) + { + this._logger.LogTrace("TextSearchProvider Input:{UserQuestion}\nOutput:{MessageText}", userQuestion, outputText); + } + } return outputText; } diff --git a/dotnet/src/Microsoft.Agents.AI/Data/TextSearchProviderOptions.cs b/dotnet/src/Microsoft.Agents.AI/TextSearchProviderOptions.cs similarity index 99% rename from dotnet/src/Microsoft.Agents.AI/Data/TextSearchProviderOptions.cs rename to dotnet/src/Microsoft.Agents.AI/TextSearchProviderOptions.cs index 6700634bcd..e90a6efa63 100644 --- a/dotnet/src/Microsoft.Agents.AI/Data/TextSearchProviderOptions.cs +++ b/dotnet/src/Microsoft.Agents.AI/TextSearchProviderOptions.cs @@ -4,7 +4,7 @@ using System.Collections.Generic; using Microsoft.Extensions.AI; -namespace Microsoft.Agents.AI.Data; +namespace Microsoft.Agents.AI; /// /// Options controlling the behavior of . diff --git a/dotnet/src/Shared/CodeTests/README.md b/dotnet/src/Shared/CodeTests/README.md new file mode 100644 index 0000000000..e1282f1778 --- /dev/null +++ b/dotnet/src/Shared/CodeTests/README.md @@ -0,0 +1,11 @@ +# Build Code + +Re-usable utility for building C# code in tests. + +To use this in your project, add the following to your `.csproj` file: + +```xml + + true + +``` diff --git a/dotnet/src/Shared/Foundry/Agents/AgentFactory.cs b/dotnet/src/Shared/Foundry/Agents/AgentFactory.cs new file mode 100644 index 0000000000..e179058e69 --- /dev/null +++ b/dotnet/src/Shared/Foundry/Agents/AgentFactory.cs @@ -0,0 +1,45 @@ +// Copyright (c) Microsoft. All rights reserved. + +#pragma warning disable IDE0005 + +using System; +using System.Threading.Tasks; +using Azure.AI.Projects; +using Azure.AI.Projects.OpenAI; + +namespace Shared.Foundry; + +internal static class AgentFactory +{ + public static async ValueTask CreateAgentAsync( + this AIProjectClient aiProjectClient, + string agentName, + AgentDefinition agentDefinition, + string agentDescription) + { + AgentVersionCreationOptions options = + new(agentDefinition) + { + Description = agentDescription, + Metadata = + { + { "deleteme", bool.TrueString }, + { "test", bool.TrueString }, + }, + }; + + AgentVersion agentVersion = await aiProjectClient.Agents.CreateAgentVersionAsync(agentName, options).ConfigureAwait(false); + + Console.ForegroundColor = ConsoleColor.Cyan; + try + { + Console.WriteLine($"PROMPT AGENT: {agentVersion.Name}:{agentVersion.Version}"); + } + finally + { + Console.ResetColor(); + } + + return agentVersion; + } +} diff --git a/dotnet/src/Shared/Foundry/Agents/README.md b/dotnet/src/Shared/Foundry/Agents/README.md new file mode 100644 index 0000000000..370068c555 --- /dev/null +++ b/dotnet/src/Shared/Foundry/Agents/README.md @@ -0,0 +1,11 @@ +# Foundry Agents + +Shared patterns for creating and utilizing Foundry agents. + +To use this in your project, add the following to your `.csproj` file: + +```xml + + true + +``` diff --git a/dotnet/src/Shared/IntegrationTests/AnthropicConfiguration.cs b/dotnet/src/Shared/IntegrationTests/AnthropicConfiguration.cs new file mode 100644 index 0000000000..2230be95ed --- /dev/null +++ b/dotnet/src/Shared/IntegrationTests/AnthropicConfiguration.cs @@ -0,0 +1,17 @@ +// Copyright (c) Microsoft. All rights reserved. + +namespace Shared.IntegrationTests; + +#pragma warning disable CS8618 // Non-nullable field must contain a non-null value when exiting constructor. Consider adding the 'required' modifier or declaring as nullable. +#pragma warning disable CA1812 // Internal class that is apparently never instantiated. + +internal sealed class AnthropicConfiguration +{ + public string? ServiceId { get; set; } + + public string ChatModelId { get; set; } + + public string ChatReasoningModelId { get; set; } + + public string ApiKey { get; set; } +} diff --git a/dotnet/src/Shared/Workflows/Execution/README.md b/dotnet/src/Shared/Workflows/Execution/README.md new file mode 100644 index 0000000000..4a885ae651 --- /dev/null +++ b/dotnet/src/Shared/Workflows/Execution/README.md @@ -0,0 +1,11 @@ +# Workflow Execution + +Common support for workflow execution. + +To use this in your project, add the following to your `.csproj` file: + +```xml + + true + +``` diff --git a/dotnet/src/Shared/Workflows/Execution/WorkflowFactory.cs b/dotnet/src/Shared/Workflows/Execution/WorkflowFactory.cs new file mode 100644 index 0000000000..08a39afe5e --- /dev/null +++ b/dotnet/src/Shared/Workflows/Execution/WorkflowFactory.cs @@ -0,0 +1,52 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Azure.Identity; +using Microsoft.Agents.AI.Workflows; +using Microsoft.Agents.AI.Workflows.Declarative; +using Microsoft.Extensions.AI; +using Microsoft.Extensions.Configuration; +using Microsoft.Extensions.Logging; +using Microsoft.Extensions.Logging.Abstractions; + +namespace Shared.Workflows; + +internal sealed class WorkflowFactory(string workflowFile, Uri foundryEndpoint) +{ + public IList Functions { get; init; } = []; + + public IConfiguration? Configuration { get; init; } + + // Assign to continue an existing conversation + public string? ConversationId { get; init; } + + // Assign to enable logging + public ILoggerFactory LoggerFactory { get; init; } = NullLoggerFactory.Instance; + + /// + /// Create the workflow from the declarative YAML. Includes definition of the + /// and the associated . + /// + public Workflow CreateWorkflow() + { + // Create the agent provider that will service agent requests within the workflow. + AzureAgentProvider agentProvider = new(foundryEndpoint, new AzureCliCredential()) + { + // Functions included here will be auto-executed by the framework. + Functions = this.Functions + }; + + // Define the workflow options. + DeclarativeWorkflowOptions options = + new(agentProvider) + { + Configuration = this.Configuration, + ConversationId = this.ConversationId, + LoggerFactory = this.LoggerFactory, + }; + + string workflowPath = Path.Combine(AppContext.BaseDirectory, workflowFile); + + // Use DeclarativeWorkflowBuilder to build a workflow based on a YAML file. + return DeclarativeWorkflowBuilder.Build(workflowPath, options); + } +} diff --git a/dotnet/src/Shared/Workflows/Execution/WorkflowRunner.cs b/dotnet/src/Shared/Workflows/Execution/WorkflowRunner.cs new file mode 100644 index 0000000000..d82bc800bb --- /dev/null +++ b/dotnet/src/Shared/Workflows/Execution/WorkflowRunner.cs @@ -0,0 +1,383 @@ +// Copyright (c) Microsoft. All rights reserved. + +// Uncomment to output unknown content types for debugging. +//#define DEBUG_OUTPUT + +using System.Diagnostics; +using System.Text.Json; +using Microsoft.Agents.AI.Workflows; +using Microsoft.Agents.AI.Workflows.Checkpointing; +using Microsoft.Agents.AI.Workflows.Declarative; +using Microsoft.Agents.AI.Workflows.Declarative.Events; +using Microsoft.Agents.AI.Workflows.Declarative.Kit; +using Microsoft.Extensions.AI; +using OpenAI.Responses; + +namespace Shared.Workflows; + +// Types are for evaluation purposes only and is subject to change or removal in future updates. +#pragma warning disable OPENAI001 +#pragma warning disable OPENAICUA001 +#pragma warning disable MEAI001 + +internal sealed class WorkflowRunner +{ + private Dictionary FunctionMap { get; } + private CheckpointInfo? LastCheckpoint { get; set; } + + public static void Notify(string message, ConsoleColor? color = null) + { + Console.ForegroundColor = color ?? ConsoleColor.Cyan; + try + { + Console.WriteLine(message); + } + finally + { + Console.ResetColor(); + } + } + + /// + /// When enabled, checkpoints will be persisted to disk as JSON files. + /// Otherwise an in-memory checkpoint store that will not persist checkpoints + /// beyond the lifetime of the process. + /// + public bool UseJsonCheckpoints { get; init; } + + public WorkflowRunner(params IEnumerable functions) + { + this.FunctionMap = functions.ToDictionary(f => f.Name); + } + + public async Task ExecuteAsync(Func workflowProvider, string input) + { + Workflow workflow = workflowProvider.Invoke(); + + CheckpointManager checkpointManager; + + if (this.UseJsonCheckpoints) + { + // Use a file-system based JSON checkpoint store to persist checkpoints to disk. + DirectoryInfo checkpointFolder = Directory.CreateDirectory(Path.Combine(".", $"chk-{DateTime.Now:yyMMdd-hhmmss-ff}")); + checkpointManager = CheckpointManager.CreateJson(new FileSystemJsonCheckpointStore(checkpointFolder)); + } + else + { + // Use an in-memory checkpoint store that will not persist checkpoints beyond the lifetime of the process. + checkpointManager = CheckpointManager.CreateInMemory(); + } + + Checkpointed run = await InProcessExecution.StreamAsync(workflow, input, checkpointManager).ConfigureAwait(false); + + bool isComplete = false; + ExternalResponse? requestResponse = null; + do + { + ExternalRequest? externalRequest = await this.MonitorAndDisposeWorkflowRunAsync(run, requestResponse).ConfigureAwait(false); + if (externalRequest is not null) + { + Notify("\nWORKFLOW: Yield\n", ConsoleColor.DarkYellow); + + if (this.LastCheckpoint is null) + { + throw new InvalidOperationException("Checkpoint information missing after external request."); + } + + // Process the external request. + object response = await this.HandleExternalRequestAsync(externalRequest).ConfigureAwait(false); + requestResponse = externalRequest.CreateResponse(response); + + // Let's resume on an entirely new workflow instance to demonstrate checkpoint portability. + workflow = workflowProvider.Invoke(); + + // Restore the latest checkpoint. + Debug.WriteLine($"RESTORE #{this.LastCheckpoint.CheckpointId}"); + Notify("WORKFLOW: Restore", ConsoleColor.DarkYellow); + + run = await InProcessExecution.ResumeStreamAsync(workflow, this.LastCheckpoint, checkpointManager, run.Run.RunId).ConfigureAwait(false); + } + else + { + isComplete = true; + } + } + while (!isComplete); + + Notify("\nWORKFLOW: Done!\n"); + } + + public async Task MonitorAndDisposeWorkflowRunAsync(Checkpointed run, ExternalResponse? response = null) + { +#pragma warning disable CA2007 // Consider calling ConfigureAwait on the awaited task + await using IAsyncDisposable disposeRun = run; +#pragma warning restore CA2007 // Consider calling ConfigureAwait on the awaited task + + bool hasStreamed = false; + string? messageId = null; + + bool shouldExit = false; + ExternalRequest? externalResponse = null; + + if (response is not null) + { + await run.Run.SendResponseAsync(response).ConfigureAwait(false); + } + + await foreach (WorkflowEvent workflowEvent in run.Run.WatchStreamAsync().ConfigureAwait(false)) + { + switch (workflowEvent) + { + case ExecutorInvokedEvent executorInvoked: + Debug.WriteLine($"EXECUTOR ENTER #{executorInvoked.ExecutorId}"); + break; + + case ExecutorCompletedEvent executorCompleted: + Debug.WriteLine($"EXECUTOR EXIT #{executorCompleted.ExecutorId}"); + break; + + case DeclarativeActionInvokedEvent actionInvoked: + Debug.WriteLine($"ACTION ENTER #{actionInvoked.ActionId} [{actionInvoked.ActionType}]"); + break; + + case DeclarativeActionCompletedEvent actionComplete: + Debug.WriteLine($"ACTION EXIT #{actionComplete.ActionId} [{actionComplete.ActionType}]"); + break; + + case ExecutorFailedEvent executorFailure: + Debug.WriteLine($"STEP ERROR #{executorFailure.ExecutorId}: {executorFailure.Data?.Message ?? "Unknown"}"); + break; + + case WorkflowErrorEvent workflowError: + throw workflowError.Data as Exception ?? new InvalidOperationException("Unexpected failure..."); + + case SuperStepCompletedEvent checkpointCompleted: + this.LastCheckpoint = checkpointCompleted.CompletionInfo?.Checkpoint; + Debug.WriteLine($"CHECKPOINT x{checkpointCompleted.StepNumber} [{this.LastCheckpoint?.CheckpointId ?? "(none)"}]"); + if (externalResponse is not null) + { + shouldExit = true; + } + break; + + case RequestInfoEvent requestInfo: + Debug.WriteLine($"REQUEST #{requestInfo.Request.RequestId}"); + externalResponse = requestInfo.Request; + break; + + case ConversationUpdateEvent invokeEvent: + Debug.WriteLine($"CONVERSATION: {invokeEvent.Data}"); + break; + + case MessageActivityEvent activityEvent: + Console.ForegroundColor = ConsoleColor.Cyan; + Console.WriteLine("\nACTIVITY:"); + Console.ForegroundColor = ConsoleColor.Yellow; + Console.WriteLine(activityEvent.Message.Trim()); + Console.ResetColor(); + break; + + case AgentRunUpdateEvent streamEvent: + if (!string.Equals(messageId, streamEvent.Update.MessageId, StringComparison.Ordinal)) + { + hasStreamed = false; + messageId = streamEvent.Update.MessageId; + + if (messageId is not null) + { + string? agentName = streamEvent.Update.AuthorName ?? streamEvent.Update.AgentId ?? nameof(ChatRole.Assistant); + Console.ForegroundColor = ConsoleColor.Cyan; + Console.Write($"\n{agentName.ToUpperInvariant()}:"); + Console.ForegroundColor = ConsoleColor.DarkGray; + Console.WriteLine($" [{messageId}]"); + Console.ResetColor(); + } + } + + ChatResponseUpdate? chatUpdate = streamEvent.Update.RawRepresentation as ChatResponseUpdate; + switch (chatUpdate?.RawRepresentation) + { + case ImageGenerationCallResponseItem messageUpdate: + await DownloadFileContentAsync(Path.GetFileName("response.png"), messageUpdate.ImageResultBytes).ConfigureAwait(false); + break; + + case FunctionCallResponseItem actionUpdate: + Console.ForegroundColor = ConsoleColor.White; + Console.Write($"Calling tool: {actionUpdate.FunctionName}"); + Console.ForegroundColor = ConsoleColor.DarkGray; + Console.WriteLine($" [{actionUpdate.CallId}]"); + Console.ResetColor(); + break; + + case McpToolCallItem actionUpdate: + Console.ForegroundColor = ConsoleColor.White; + Console.Write($"Calling tool: {actionUpdate.ToolName}"); + Console.ForegroundColor = ConsoleColor.DarkGray; + Console.WriteLine($" [{actionUpdate.Id}]"); + Console.ResetColor(); + break; + } + + try + { + Console.ResetColor(); + Console.Write(streamEvent.Update.Text); + hasStreamed |= !string.IsNullOrEmpty(streamEvent.Update.Text); + } + finally + { + Console.ResetColor(); + } + break; + + case AgentRunResponseEvent messageEvent: + try + { + if (hasStreamed) + { + Console.WriteLine(); + } + + if (messageEvent.Response.Usage is not null) + { + Console.ForegroundColor = ConsoleColor.DarkGray; + Console.WriteLine($"[Tokens Total: {messageEvent.Response.Usage.TotalTokenCount}, Input: {messageEvent.Response.Usage.InputTokenCount}, Output: {messageEvent.Response.Usage.OutputTokenCount}]"); + Console.ResetColor(); + } + } + finally + { + Console.ResetColor(); + } + break; + + default: +#if DEBUG_OUTPUT + Debug.WriteLine($"UNHANDLED: {workflowEvent.GetType().Name}"); +#endif + break; + } + + if (shouldExit) + { + break; + } + } + + return externalResponse; + } + + /// + /// Handle request for external input. + /// + private async ValueTask HandleExternalRequestAsync(ExternalRequest request) + { + ExternalInputRequest inputRequest = + request.DataAs() ?? + throw new InvalidOperationException($"Expected external request type: {request.GetType().Name}."); + + List responseMessages = []; + + foreach (ChatMessage message in inputRequest.AgentResponse.Messages) + { + await foreach (ChatMessage responseMessage in this.ProcessInputMessageAsync(message).ConfigureAwait(false)) + { + responseMessages.Add(responseMessage); + } + } + + if (responseMessages.Count == 0) + { + // Must be request for user input. + responseMessages.Add(HandleUserInputRequest(inputRequest)); + } + + Console.WriteLine(); + + return new ExternalInputResponse(responseMessages); + } + + private async IAsyncEnumerable ProcessInputMessageAsync(ChatMessage message) + { + foreach (AIContent requestItem in message.Contents) + { + ChatMessage? responseMessage = + requestItem switch + { + FunctionCallContent functionCall => await InvokeFunctionAsync(functionCall).ConfigureAwait(false), + FunctionApprovalRequestContent functionApprovalRequest => ApproveFunction(functionApprovalRequest), + McpServerToolApprovalRequestContent mcpApprovalRequest => ApproveMCP(mcpApprovalRequest), + _ => HandleUnknown(requestItem), + }; + + if (responseMessage is not null) + { + yield return responseMessage; + } + } + + ChatMessage? HandleUnknown(AIContent request) + { +#if DEBUG_OUTPUT + Notify($"INPUT - Unknown: {request.GetType().Name} [{request.RawRepresentation?.GetType().Name ?? "*"}]"); +#endif + return null; + } + + ChatMessage ApproveFunction(FunctionApprovalRequestContent functionApprovalRequest) + { + Notify($"INPUT - Approving Function: {functionApprovalRequest.FunctionCall.Name}"); + return new ChatMessage(ChatRole.User, [functionApprovalRequest.CreateResponse(approved: true)]); + } + + ChatMessage ApproveMCP(McpServerToolApprovalRequestContent mcpApprovalRequest) + { + Notify($"INPUT - Approving MCP: {mcpApprovalRequest.ToolCall.ToolName}"); + return new ChatMessage(ChatRole.User, [mcpApprovalRequest.CreateResponse(approved: true)]); + } + + async Task InvokeFunctionAsync(FunctionCallContent functionCall) + { + Notify($"INPUT - Executing Function: {functionCall.Name}"); + AIFunction functionTool = this.FunctionMap[functionCall.Name]; + AIFunctionArguments? functionArguments = functionCall.Arguments is null ? null : new(functionCall.Arguments.NormalizePortableValues()); + object? result = await functionTool.InvokeAsync(functionArguments).ConfigureAwait(false); + return new ChatMessage(ChatRole.Tool, [new FunctionResultContent(functionCall.CallId, JsonSerializer.Serialize(result))]); + } + } + + private static ChatMessage HandleUserInputRequest(ExternalInputRequest request) + { + string prompt = + string.IsNullOrWhiteSpace(request.AgentResponse.Text) || request.AgentResponse.ResponseId is not null ? + "INPUT:" : + request.AgentResponse.Text; + + string? userInput; + do + { + Console.ForegroundColor = ConsoleColor.DarkGreen; + Console.Write($"{prompt} "); + Console.ForegroundColor = ConsoleColor.White; + userInput = Console.ReadLine(); + } + while (string.IsNullOrWhiteSpace(userInput)); + + return new ChatMessage(ChatRole.User, userInput); + } + + private static async ValueTask DownloadFileContentAsync(string filename, BinaryData content) + { + string filePath = Path.Combine(Path.GetTempPath(), Path.GetFileName(filename)); + filePath = Path.ChangeExtension(filePath, ".png"); + + await File.WriteAllBytesAsync(filePath, content.ToArray()).ConfigureAwait(false); + + Process.Start( + new ProcessStartInfo + { + FileName = "cmd.exe", + Arguments = $"/C start {filePath}" + }); + } +} diff --git a/dotnet/src/Shared/Workflows/Settings/Application.cs b/dotnet/src/Shared/Workflows/Settings/Application.cs new file mode 100644 index 0000000000..de8eb51534 --- /dev/null +++ b/dotnet/src/Shared/Workflows/Settings/Application.cs @@ -0,0 +1,80 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Reflection; +using Microsoft.Extensions.Configuration; + +namespace Shared.Workflows; + +internal static class Application +{ + /// + /// Configuration key used to identify the Foundry project endpoint. + /// + public static class Settings + { + public const string FoundryEndpoint = "FOUNDRY_PROJECT_ENDPOINT"; + public const string FoundryModelMini = "FOUNDRY_MODEL_DEPLOYMENT_NAME"; + public const string FoundryModelFull = "FOUNDRY_MEDIA_DEPLOYMENT_NAME"; + public const string FoundryGroundingTool = "FOUNDRY_CONNECTION_GROUNDING_TOOL"; + } + + public static string GetInput(string[] args) + { + string? input = args.FirstOrDefault(); + + try + { + Console.ForegroundColor = ConsoleColor.DarkGreen; + + Console.Write("\nINPUT: "); + + Console.ForegroundColor = ConsoleColor.White; + + if (!string.IsNullOrWhiteSpace(input)) + { + Console.WriteLine(input); + return input; + } + while (string.IsNullOrWhiteSpace(input)) + { + input = Console.ReadLine(); + } + + return input.Trim(); + } + finally + { + Console.ResetColor(); + } + } + + public static string? GetRepoFolder() + { + DirectoryInfo? current = new(Directory.GetCurrentDirectory()); + + while (current is not null) + { + if (Directory.Exists(Path.Combine(current.FullName, ".git"))) + { + return current.FullName; + } + + current = current.Parent; + } + + return null; + } + + public static string GetValue(this IConfiguration configuration, string settingName) => + configuration[settingName] ?? + throw new InvalidOperationException($"Undefined configuration setting: {settingName}"); + + /// + /// Initialize configuration and environment + /// + public static IConfigurationRoot InitializeConfig() => + new ConfigurationBuilder() + .AddUserSecrets(Assembly.GetExecutingAssembly()) + .AddEnvironmentVariables() + .Build(); +} diff --git a/dotnet/src/Shared/Workflows/Settings/README.md b/dotnet/src/Shared/Workflows/Settings/README.md new file mode 100644 index 0000000000..80b176131b --- /dev/null +++ b/dotnet/src/Shared/Workflows/Settings/README.md @@ -0,0 +1,11 @@ +# Workflow Settings + +Common support configuration and environment used in workflow samples. + +To use this in your project, add the following to your `.csproj` file: + +```xml + + true + +``` diff --git a/dotnet/tests/.editorconfig b/dotnet/tests/.editorconfig index e3ee57d9ba..a200bbb9ed 100644 --- a/dotnet/tests/.editorconfig +++ b/dotnet/tests/.editorconfig @@ -1,6 +1,10 @@ # Suppressing errors for Test projects under dotnet/tests folder [*.cs] +dotnet_diagnostic.CA1822.severity = none # Member does not access instance data and can be marked as static +dotnet_diagnostic.CA1873.severity = none # Evaluation of logging arguments may be expensive +dotnet_diagnostic.CA1875.severity = none # Regex.IsMatch/Count instead of Regex.Match(...).Success/Regex.Matches(...).Count dotnet_diagnostic.CA2007.severity = none # Do not directly await a Task +dotnet_diagnostic.CA2249.severity = none # Use `string.Contains` instead of `string.IndexOf` to improve readability dotnet_diagnostic.CS1591.severity = none # Missing XML comment for publicly visible type or member diff --git a/dotnet/samples/.gitignore b/dotnet/tests/.gitignore similarity index 100% rename from dotnet/samples/.gitignore rename to dotnet/tests/.gitignore diff --git a/dotnet/tests/AgentConformance.IntegrationTests/AgentConformance.IntegrationTests.csproj b/dotnet/tests/AgentConformance.IntegrationTests/AgentConformance.IntegrationTests.csproj index 90347f3ce8..5ac895d63c 100644 --- a/dotnet/tests/AgentConformance.IntegrationTests/AgentConformance.IntegrationTests.csproj +++ b/dotnet/tests/AgentConformance.IntegrationTests/AgentConformance.IntegrationTests.csproj @@ -1,7 +1,6 @@ - $(ProjectsTargetFrameworks) false @@ -11,7 +10,10 @@ - + + + + diff --git a/dotnet/tests/AgentConformance.IntegrationTests/RunStreamingTests.cs b/dotnet/tests/AgentConformance.IntegrationTests/RunStreamingTests.cs index 984356affb..a2da3e0d6e 100644 --- a/dotnet/tests/AgentConformance.IntegrationTests/RunStreamingTests.cs +++ b/dotnet/tests/AgentConformance.IntegrationTests/RunStreamingTests.cs @@ -4,6 +4,7 @@ using System.Linq; using System.Threading.Tasks; using AgentConformance.IntegrationTests.Support; +using Microsoft.Agents.AI; using Microsoft.Extensions.AI; namespace AgentConformance.IntegrationTests; @@ -16,6 +17,8 @@ namespace AgentConformance.IntegrationTests; public abstract class RunStreamingTests(Func createAgentFixture) : AgentTests(createAgentFixture) where TAgentFixture : IAgentFixture { + public virtual Func> AgentRunOptionsFactory { get; set; } = () => Task.FromResult(default(AgentRunOptions)); + [RetryFact(Constants.RetryCount, Constants.RetryDelay)] public virtual async Task RunWithNoMessageDoesNotFailAsync() { @@ -25,7 +28,7 @@ public virtual async Task RunWithNoMessageDoesNotFailAsync() await using var cleanup = new ThreadCleanup(thread, this.Fixture); // Act - var chatResponses = await agent.RunStreamingAsync(thread).ToListAsync(); + var chatResponses = await agent.RunStreamingAsync(thread, await this.AgentRunOptionsFactory.Invoke()).ToListAsync(); } [RetryFact(Constants.RetryCount, Constants.RetryDelay)] @@ -37,7 +40,7 @@ public virtual async Task RunWithStringReturnsExpectedResultAsync() await using var cleanup = new ThreadCleanup(thread, this.Fixture); // Act - var responseUpdates = await agent.RunStreamingAsync("What is the capital of France.", thread).ToListAsync(); + var responseUpdates = await agent.RunStreamingAsync("What is the capital of France.", thread, await this.AgentRunOptionsFactory.Invoke()).ToListAsync(); // Assert var chatResponseText = string.Concat(responseUpdates.Select(x => x.Text)); @@ -53,7 +56,7 @@ public virtual async Task RunWithChatMessageReturnsExpectedResultAsync() await using var cleanup = new ThreadCleanup(thread, this.Fixture); // Act - var responseUpdates = await agent.RunStreamingAsync(new ChatMessage(ChatRole.User, "What is the capital of France."), thread).ToListAsync(); + var responseUpdates = await agent.RunStreamingAsync(new ChatMessage(ChatRole.User, "What is the capital of France."), thread, await this.AgentRunOptionsFactory.Invoke()).ToListAsync(); // Assert var chatResponseText = string.Concat(responseUpdates.Select(x => x.Text)); @@ -74,7 +77,8 @@ public virtual async Task RunWithChatMessagesReturnsExpectedResultAsync() new ChatMessage(ChatRole.User, "Hello."), new ChatMessage(ChatRole.User, "What is the capital of France.") ], - thread).ToListAsync(); + thread, + await this.AgentRunOptionsFactory.Invoke()).ToListAsync(); // Assert var chatResponseText = string.Concat(responseUpdates.Select(x => x.Text)); @@ -92,8 +96,9 @@ public virtual async Task ThreadMaintainsHistoryAsync() await using var cleanup = new ThreadCleanup(thread, this.Fixture); // Act - var responseUpdates1 = await agent.RunStreamingAsync(Q1, thread).ToListAsync(); - var responseUpdates2 = await agent.RunStreamingAsync(Q2, thread).ToListAsync(); + var options = await this.AgentRunOptionsFactory.Invoke(); + var responseUpdates1 = await agent.RunStreamingAsync(Q1, thread, options).ToListAsync(); + var responseUpdates2 = await agent.RunStreamingAsync(Q2, thread, options).ToListAsync(); // Assert var response1Text = string.Concat(responseUpdates1.Select(x => x.Text)); diff --git a/dotnet/tests/AgentConformance.IntegrationTests/RunTests.cs b/dotnet/tests/AgentConformance.IntegrationTests/RunTests.cs index f89c821455..58f8b67d1d 100644 --- a/dotnet/tests/AgentConformance.IntegrationTests/RunTests.cs +++ b/dotnet/tests/AgentConformance.IntegrationTests/RunTests.cs @@ -4,6 +4,7 @@ using System.Linq; using System.Threading.Tasks; using AgentConformance.IntegrationTests.Support; +using Microsoft.Agents.AI; using Microsoft.Extensions.AI; namespace AgentConformance.IntegrationTests; @@ -16,6 +17,8 @@ namespace AgentConformance.IntegrationTests; public abstract class RunTests(Func createAgentFixture) : AgentTests(createAgentFixture) where TAgentFixture : IAgentFixture { + public virtual Func> AgentRunOptionsFactory { get; set; } = () => Task.FromResult(default(AgentRunOptions)); + [RetryFact(Constants.RetryCount, Constants.RetryDelay)] public virtual async Task RunWithNoMessageDoesNotFailAsync() { @@ -40,7 +43,7 @@ public virtual async Task RunWithStringReturnsExpectedResultAsync() await using var cleanup = new ThreadCleanup(thread, this.Fixture); // Act - var response = await agent.RunAsync("What is the capital of France.", thread); + var response = await agent.RunAsync("What is the capital of France.", thread, await this.AgentRunOptionsFactory.Invoke()); // Assert Assert.NotNull(response); @@ -58,7 +61,7 @@ public virtual async Task RunWithChatMessageReturnsExpectedResultAsync() await using var cleanup = new ThreadCleanup(thread, this.Fixture); // Act - var response = await agent.RunAsync(new ChatMessage(ChatRole.User, "What is the capital of France."), thread); + var response = await agent.RunAsync(new ChatMessage(ChatRole.User, "What is the capital of France."), thread, await this.AgentRunOptionsFactory.Invoke()); // Assert Assert.NotNull(response); @@ -80,7 +83,8 @@ public virtual async Task RunWithChatMessagesReturnsExpectedResultAsync() new ChatMessage(ChatRole.User, "Hello."), new ChatMessage(ChatRole.User, "What is the capital of France.") ], - thread); + thread, + await this.AgentRunOptionsFactory.Invoke()); // Assert Assert.NotNull(response); @@ -99,8 +103,9 @@ public virtual async Task ThreadMaintainsHistoryAsync() await using var cleanup = new ThreadCleanup(thread, this.Fixture); // Act - var result1 = await agent.RunAsync(Q1, thread); - var result2 = await agent.RunAsync(Q2, thread); + var options = await this.AgentRunOptionsFactory.Invoke(); + var result1 = await agent.RunAsync(Q1, thread, options); + var result2 = await agent.RunAsync(Q2, thread, options); // Assert Assert.Contains("Paris", result1.Text); @@ -111,8 +116,8 @@ public virtual async Task ThreadMaintainsHistoryAsync() Assert.Equal(2, chatHistory.Count(x => x.Role == ChatRole.User)); Assert.Equal(2, chatHistory.Count(x => x.Role == ChatRole.Assistant)); Assert.Equal(Q1, chatHistory[0].Text); - Assert.Equal(Q2, chatHistory[2].Text); Assert.Contains("Paris", chatHistory[1].Text); + Assert.Equal(Q2, chatHistory[2].Text); Assert.Contains("Vienna", chatHistory[3].Text); } } diff --git a/dotnet/tests/AnthropicChatCompletion.IntegrationTests/AnthropicChatCompletion.IntegrationTests.csproj b/dotnet/tests/AnthropicChatCompletion.IntegrationTests/AnthropicChatCompletion.IntegrationTests.csproj new file mode 100644 index 0000000000..929eafe998 --- /dev/null +++ b/dotnet/tests/AnthropicChatCompletion.IntegrationTests/AnthropicChatCompletion.IntegrationTests.csproj @@ -0,0 +1,20 @@ + + + + True + + + + + + + + + + + + + + + + diff --git a/dotnet/tests/AnthropicChatCompletion.IntegrationTests/AnthropicChatCompletionChatClientAgentRunStreamingTests.cs b/dotnet/tests/AnthropicChatCompletion.IntegrationTests/AnthropicChatCompletionChatClientAgentRunStreamingTests.cs new file mode 100644 index 0000000000..992db5380b --- /dev/null +++ b/dotnet/tests/AnthropicChatCompletion.IntegrationTests/AnthropicChatCompletionChatClientAgentRunStreamingTests.cs @@ -0,0 +1,26 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Threading.Tasks; +using AgentConformance.IntegrationTests; + +namespace AnthropicChatCompletion.IntegrationTests; + +public abstract class SkipAllChatClientRunStreaming(Func func) : ChatClientAgentRunStreamingTests(func) +{ + [Fact(Skip = AnthropicChatCompletionFixture.SkipReason)] + public override Task RunWithFunctionsInvokesFunctionsAndReturnsExpectedResultsAsync() + => base.RunWithFunctionsInvokesFunctionsAndReturnsExpectedResultsAsync(); + + [Fact(Skip = AnthropicChatCompletionFixture.SkipReason)] + public override Task RunWithInstructionsAndNoMessageReturnsExpectedResultAsync() + => base.RunWithInstructionsAndNoMessageReturnsExpectedResultAsync(); +} + +public class AnthropicBetaChatCompletionChatClientAgentReasoningRunStreamingTests() : SkipAllChatClientRunStreaming(() => new(useReasoningChatModel: true, useBeta: true)); + +public class AnthropicBetaChatCompletionChatClientAgentRunStreamingTests() : SkipAllChatClientRunStreaming(() => new(useReasoningChatModel: false, useBeta: true)); + +public class AnthropicChatCompletionChatClientAgentRunStreamingTests() : SkipAllChatClientRunStreaming(() => new(useReasoningChatModel: false, useBeta: false)); + +public class AnthropicChatCompletionChatClientAgentReasoningRunStreamingTests() : SkipAllChatClientRunStreaming(() => new(useReasoningChatModel: true, useBeta: false)); diff --git a/dotnet/tests/AnthropicChatCompletion.IntegrationTests/AnthropicChatCompletionChatClientAgentRunTests.cs b/dotnet/tests/AnthropicChatCompletion.IntegrationTests/AnthropicChatCompletionChatClientAgentRunTests.cs new file mode 100644 index 0000000000..e2ce6e5d04 --- /dev/null +++ b/dotnet/tests/AnthropicChatCompletion.IntegrationTests/AnthropicChatCompletionChatClientAgentRunTests.cs @@ -0,0 +1,30 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Threading.Tasks; +using AgentConformance.IntegrationTests; + +namespace AnthropicChatCompletion.IntegrationTests; + +public abstract class SkipAllChatClientAgentRun(Func func) : ChatClientAgentRunTests(func) +{ + [Fact(Skip = AnthropicChatCompletionFixture.SkipReason)] + public override Task RunWithFunctionsInvokesFunctionsAndReturnsExpectedResultsAsync() + => base.RunWithFunctionsInvokesFunctionsAndReturnsExpectedResultsAsync(); + + [Fact(Skip = AnthropicChatCompletionFixture.SkipReason)] + public override Task RunWithInstructionsAndNoMessageReturnsExpectedResultAsync() + => base.RunWithInstructionsAndNoMessageReturnsExpectedResultAsync(); +} + +public class AnthropicBetaChatCompletionChatClientAgentRunTests() + : SkipAllChatClientAgentRun(() => new(useReasoningChatModel: false, useBeta: true)); + +public class AnthropicBetaChatCompletionChatClientAgentReasoningRunTests() + : SkipAllChatClientAgentRun(() => new(useReasoningChatModel: true, useBeta: true)); + +public class AnthropicChatCompletionChatClientAgentRunTests() + : SkipAllChatClientAgentRun(() => new(useReasoningChatModel: false, useBeta: false)); + +public class AnthropicChatCompletionChatClientAgentReasoningRunTests() + : SkipAllChatClientAgentRun(() => new(useReasoningChatModel: true, useBeta: false)); diff --git a/dotnet/tests/AnthropicChatCompletion.IntegrationTests/AnthropicChatCompletionFixture.cs b/dotnet/tests/AnthropicChatCompletion.IntegrationTests/AnthropicChatCompletionFixture.cs new file mode 100644 index 0000000000..72c0b14ae2 --- /dev/null +++ b/dotnet/tests/AnthropicChatCompletion.IntegrationTests/AnthropicChatCompletionFixture.cs @@ -0,0 +1,104 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Collections.Generic; +using System.Linq; +using System.Threading.Tasks; +using AgentConformance.IntegrationTests; +using AgentConformance.IntegrationTests.Support; +using Anthropic; +using Anthropic.Models.Beta.Messages; +using Anthropic.Models.Messages; +using Microsoft.Agents.AI; +using Microsoft.Extensions.AI; +using Shared.IntegrationTests; + +namespace AnthropicChatCompletion.IntegrationTests; + +public class AnthropicChatCompletionFixture : IChatClientAgentFixture +{ + // All tests for Anthropic are intended to be ran locally as the CI pipeline for Anthropic is not setup. + internal const string SkipReason = "Integrations tests for local execution only"; + + private static readonly AnthropicConfiguration s_config = TestConfiguration.LoadSection(); + private readonly bool _useReasoningModel; + private readonly bool _useBeta; + + private ChatClientAgent _agent = null!; + + public AnthropicChatCompletionFixture(bool useReasoningChatModel, bool useBeta) + { + this._useReasoningModel = useReasoningChatModel; + this._useBeta = useBeta; + } + + public AIAgent Agent => this._agent; + + public IChatClient ChatClient => this._agent.ChatClient; + + public async Task> GetChatHistoryAsync(AgentThread thread) + { + var typedThread = (ChatClientAgentThread)thread; + + return typedThread.MessageStore is null ? [] : (await typedThread.MessageStore.GetMessagesAsync()).ToList(); + } + + public Task CreateChatClientAgentAsync( + string name = "HelpfulAssistant", + string instructions = "You are a helpful assistant.", + IList? aiTools = null) + { + var anthropicClient = new AnthropicClient() { APIKey = s_config.ApiKey }; + + IChatClient? chatClient = this._useBeta + ? anthropicClient + .Beta + .AsIChatClient() + .AsBuilder() + .ConfigureOptions(options + => options.RawRepresentationFactory = _ + => new Anthropic.Models.Beta.Messages.MessageCreateParams() + { + Model = options.ModelId ?? (this._useReasoningModel ? s_config.ChatReasoningModelId : s_config.ChatModelId), + MaxTokens = options.MaxOutputTokens ?? 4096, + Messages = [], + Thinking = this._useReasoningModel + ? new BetaThinkingConfigParam(new BetaThinkingConfigEnabled(2048)) + : new BetaThinkingConfigParam(new BetaThinkingConfigDisabled()) + }).Build() + + : anthropicClient + .AsIChatClient() + .AsBuilder() + .ConfigureOptions(options + => options.RawRepresentationFactory = _ + => new Anthropic.Models.Messages.MessageCreateParams() + { + Model = options.ModelId ?? (this._useReasoningModel ? s_config.ChatReasoningModelId : s_config.ChatModelId), + MaxTokens = options.MaxOutputTokens ?? 4096, + Messages = [], + Thinking = this._useReasoningModel + ? new ThinkingConfigParam(new ThinkingConfigEnabled(2048)) + : new ThinkingConfigParam(new ThinkingConfigDisabled()) + }).Build(); + + return Task.FromResult(new ChatClientAgent(chatClient, options: new() + { + Name = name, + ChatOptions = new() { Instructions = instructions, Tools = aiTools } + })); + } + + public Task DeleteAgentAsync(ChatClientAgent agent) => + // Chat Completion does not require/support deleting agents, so this is a no-op. + Task.CompletedTask; + + public Task DeleteThreadAsync(AgentThread thread) => + // Chat Completion does not require/support deleting threads, so this is a no-op. + Task.CompletedTask; + + public async Task InitializeAsync() => + this._agent = await this.CreateChatClientAgentAsync(); + + public Task DisposeAsync() => + Task.CompletedTask; +} diff --git a/dotnet/tests/AnthropicChatCompletion.IntegrationTests/AnthropicChatCompletionRunStreamingTests.cs b/dotnet/tests/AnthropicChatCompletion.IntegrationTests/AnthropicChatCompletionRunStreamingTests.cs new file mode 100644 index 0000000000..f1bbbe47e9 --- /dev/null +++ b/dotnet/tests/AnthropicChatCompletion.IntegrationTests/AnthropicChatCompletionRunStreamingTests.cs @@ -0,0 +1,37 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Threading.Tasks; +using AgentConformance.IntegrationTests; + +namespace AnthropicChatCompletion.IntegrationTests; + +public abstract class SkipAllRunStreaming(Func func) : RunStreamingTests(func) +{ + [Fact(Skip = AnthropicChatCompletionFixture.SkipReason)] + public override Task RunWithChatMessageReturnsExpectedResultAsync() => base.RunWithChatMessageReturnsExpectedResultAsync(); + + [Fact(Skip = AnthropicChatCompletionFixture.SkipReason)] + public override Task RunWithNoMessageDoesNotFailAsync() => base.RunWithNoMessageDoesNotFailAsync(); + + [Fact(Skip = AnthropicChatCompletionFixture.SkipReason)] + public override Task RunWithChatMessagesReturnsExpectedResultAsync() => base.RunWithChatMessagesReturnsExpectedResultAsync(); + + [Fact(Skip = AnthropicChatCompletionFixture.SkipReason)] + public override Task RunWithStringReturnsExpectedResultAsync() => base.RunWithStringReturnsExpectedResultAsync(); + + [Fact(Skip = AnthropicChatCompletionFixture.SkipReason)] + public override Task ThreadMaintainsHistoryAsync() => base.ThreadMaintainsHistoryAsync(); +} + +public class AnthropicBetaChatCompletionRunStreamingTests() + : SkipAllRunStreaming(() => new(useReasoningChatModel: false, useBeta: true)); + +public class AnthropicBetaChatCompletionReasoningRunStreamingTests() + : SkipAllRunStreaming(() => new(useReasoningChatModel: true, useBeta: true)); + +public class AnthropicChatCompletionRunStreamingTests() + : SkipAllRunStreaming(() => new(useReasoningChatModel: false, useBeta: false)); + +public class AnthropicChatCompletionReasoningRunStreamingTests() + : SkipAllRunStreaming(() => new(useReasoningChatModel: true, useBeta: false)); diff --git a/dotnet/tests/AnthropicChatCompletion.IntegrationTests/AnthropicChatCompletionRunTests.cs b/dotnet/tests/AnthropicChatCompletion.IntegrationTests/AnthropicChatCompletionRunTests.cs new file mode 100644 index 0000000000..aadbf747c2 --- /dev/null +++ b/dotnet/tests/AnthropicChatCompletion.IntegrationTests/AnthropicChatCompletionRunTests.cs @@ -0,0 +1,37 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Threading.Tasks; +using AgentConformance.IntegrationTests; + +namespace AnthropicChatCompletion.IntegrationTests; + +public abstract class SkipAllRun(Func func) : RunTests(func) +{ + [Fact(Skip = AnthropicChatCompletionFixture.SkipReason)] + public override Task RunWithChatMessageReturnsExpectedResultAsync() => base.RunWithChatMessageReturnsExpectedResultAsync(); + + [Fact(Skip = AnthropicChatCompletionFixture.SkipReason)] + public override Task RunWithNoMessageDoesNotFailAsync() => base.RunWithNoMessageDoesNotFailAsync(); + + [Fact(Skip = AnthropicChatCompletionFixture.SkipReason)] + public override Task RunWithChatMessagesReturnsExpectedResultAsync() => base.RunWithChatMessagesReturnsExpectedResultAsync(); + + [Fact(Skip = AnthropicChatCompletionFixture.SkipReason)] + public override Task RunWithStringReturnsExpectedResultAsync() => base.RunWithStringReturnsExpectedResultAsync(); + + [Fact(Skip = AnthropicChatCompletionFixture.SkipReason)] + public override Task ThreadMaintainsHistoryAsync() => base.ThreadMaintainsHistoryAsync(); +} + +public class AnthropicBetaChatCompletionRunTests() + : SkipAllRun(() => new(useReasoningChatModel: false, useBeta: true)); + +public class AnthropicBetaChatCompletionReasoningRunTests() + : SkipAllRun(() => new(useReasoningChatModel: true, useBeta: true)); + +public class AnthropicChatCompletionRunTests() + : SkipAllRun(() => new(useReasoningChatModel: false, useBeta: false)); + +public class AnthropicChatCompletionReasoningRunTests() + : SkipAllRun(() => new(useReasoningChatModel: true, useBeta: false)); diff --git a/dotnet/tests/AzureAI.IntegrationTests/AIProjectClientAgentRunStreamingTests.cs b/dotnet/tests/AzureAI.IntegrationTests/AIProjectClientAgentRunStreamingTests.cs new file mode 100644 index 0000000000..50ced1e64d --- /dev/null +++ b/dotnet/tests/AzureAI.IntegrationTests/AIProjectClientAgentRunStreamingTests.cs @@ -0,0 +1,32 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Threading.Tasks; +using AgentConformance.IntegrationTests; +using Microsoft.Agents.AI; + +namespace AzureAI.IntegrationTests; + +public class AIProjectClientAgentRunStreamingPreviousResponseTests() : RunStreamingTests(() => new()) +{ + [Fact(Skip = "No messages is not supported")] + public override Task RunWithNoMessageDoesNotFailAsync() + { + return Task.CompletedTask; + } +} + +public class AIProjectClientAgentRunStreamingConversationTests() : RunTests(() => new()) +{ + public override Func> AgentRunOptionsFactory => async () => + { + var conversationId = await this.Fixture.CreateConversationAsync(); + return new ChatClientAgentRunOptions(new() { ConversationId = conversationId }); + }; + + [Fact(Skip = "No messages is not supported")] + public override Task RunWithNoMessageDoesNotFailAsync() + { + return Task.CompletedTask; + } +} diff --git a/dotnet/tests/AzureAI.IntegrationTests/AIProjectClientAgentRunTests.cs b/dotnet/tests/AzureAI.IntegrationTests/AIProjectClientAgentRunTests.cs new file mode 100644 index 0000000000..0092090401 --- /dev/null +++ b/dotnet/tests/AzureAI.IntegrationTests/AIProjectClientAgentRunTests.cs @@ -0,0 +1,32 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Threading.Tasks; +using AgentConformance.IntegrationTests; +using Microsoft.Agents.AI; + +namespace AzureAI.IntegrationTests; + +public class AIProjectClientAgentRunPreviousResponseTests() : RunTests(() => new()) +{ + [Fact(Skip = "No messages is not supported")] + public override Task RunWithNoMessageDoesNotFailAsync() + { + return Task.CompletedTask; + } +} + +public class AIProjectClientAgentRunConversationTests() : RunTests(() => new()) +{ + public override Func> AgentRunOptionsFactory => async () => + { + var conversationId = await this.Fixture.CreateConversationAsync(); + return new ChatClientAgentRunOptions(new() { ConversationId = conversationId }); + }; + + [Fact(Skip = "No messages is not supported")] + public override Task RunWithNoMessageDoesNotFailAsync() + { + return Task.CompletedTask; + } +} diff --git a/dotnet/tests/AzureAI.IntegrationTests/AIProjectClientChatClientAgentRunStreamingTests.cs b/dotnet/tests/AzureAI.IntegrationTests/AIProjectClientChatClientAgentRunStreamingTests.cs new file mode 100644 index 0000000000..befa409d80 --- /dev/null +++ b/dotnet/tests/AzureAI.IntegrationTests/AIProjectClientChatClientAgentRunStreamingTests.cs @@ -0,0 +1,15 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Threading.Tasks; +using AgentConformance.IntegrationTests; + +namespace AzureAI.IntegrationTests; + +public class AIProjectClientChatClientAgentRunStreamingTests() : ChatClientAgentRunStreamingTests(() => new()) +{ + [Fact(Skip = "No messages is not supported")] + public override Task RunWithInstructionsAndNoMessageReturnsExpectedResultAsync() + { + return Task.CompletedTask; + } +} diff --git a/dotnet/tests/AzureAI.IntegrationTests/AIProjectClientChatClientAgentRunTests.cs b/dotnet/tests/AzureAI.IntegrationTests/AIProjectClientChatClientAgentRunTests.cs new file mode 100644 index 0000000000..1af12606cb --- /dev/null +++ b/dotnet/tests/AzureAI.IntegrationTests/AIProjectClientChatClientAgentRunTests.cs @@ -0,0 +1,15 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Threading.Tasks; +using AgentConformance.IntegrationTests; + +namespace AzureAI.IntegrationTests; + +public class AIProjectClientChatClientAgentRunTests() : ChatClientAgentRunTests(() => new()) +{ + [Fact(Skip = "No messages is not supported")] + public override Task RunWithInstructionsAndNoMessageReturnsExpectedResultAsync() + { + return Task.CompletedTask; + } +} diff --git a/dotnet/tests/AzureAI.IntegrationTests/AIProjectClientCreateTests.cs b/dotnet/tests/AzureAI.IntegrationTests/AIProjectClientCreateTests.cs new file mode 100644 index 0000000000..f626736418 --- /dev/null +++ b/dotnet/tests/AzureAI.IntegrationTests/AIProjectClientCreateTests.cs @@ -0,0 +1,277 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.IO; +using System.Threading.Tasks; +using AgentConformance.IntegrationTests.Support; +using Azure.AI.Projects; +using Azure.AI.Projects.OpenAI; +using Azure.Identity; +using Microsoft.Agents.AI; +using Microsoft.Extensions.AI; +using OpenAI.Files; +using OpenAI.Responses; +using Shared.IntegrationTests; + +namespace AzureAI.IntegrationTests; + +public class AIProjectClientCreateTests +{ + private static readonly AzureAIConfiguration s_config = TestConfiguration.LoadSection(); + private readonly AIProjectClient _client = new(new Uri(s_config.Endpoint), new AzureCliCredential()); + + [Theory] + [InlineData("CreateWithChatClientAgentOptionsAsync")] + [InlineData("CreateWithChatClientAgentOptionsSync")] + [InlineData("CreateWithFoundryOptionsAsync")] + [InlineData("CreateWithFoundryOptionsSync")] + public async Task CreateAgent_CreatesAgentWithCorrectMetadataAsync(string createMechanism) + { + // Arrange. + string AgentName = AIProjectClientFixture.GenerateUniqueAgentName("IntegrationTestAgent"); + const string AgentDescription = "An agent created during integration tests"; + const string AgentInstructions = "You are an integration test agent"; + + // Act. + var agent = createMechanism switch + { + "CreateWithChatClientAgentOptionsAsync" => await this._client.CreateAIAgentAsync( + model: s_config.DeploymentName, + options: new ChatClientAgentOptions() + { + Name = AgentName, + Description = AgentDescription, + ChatOptions = new() { Instructions = AgentInstructions } + }), + "CreateWithChatClientAgentOptionsSync" => this._client.CreateAIAgent( + model: s_config.DeploymentName, + options: new ChatClientAgentOptions() + { + Name = AgentName, + Description = AgentDescription, + ChatOptions = new() { Instructions = AgentInstructions } + }), + "CreateWithFoundryOptionsAsync" => await this._client.CreateAIAgentAsync( + name: AgentName, + creationOptions: new AgentVersionCreationOptions(new PromptAgentDefinition(s_config.DeploymentName) { Instructions = AgentInstructions }) { Description = AgentDescription }), + "CreateWithFoundryOptionsSync" => this._client.CreateAIAgent( + name: AgentName, + creationOptions: new AgentVersionCreationOptions(new PromptAgentDefinition(s_config.DeploymentName) { Instructions = AgentInstructions }) { Description = AgentDescription }), + _ => throw new InvalidOperationException($"Unknown create mechanism: {createMechanism}") + }; + + try + { + // Assert. + Assert.NotNull(agent); + Assert.Equal(AgentName, agent.Name); + Assert.Equal(AgentDescription, agent.Description); + Assert.Equal(AgentInstructions, agent.Instructions); + + var agentRecord = await this._client.Agents.GetAgentAsync(agent.Name); + Assert.NotNull(agentRecord); + Assert.Equal(AgentName, agentRecord.Value.Name); + var definition = Assert.IsType(agentRecord.Value.Versions.Latest.Definition); + Assert.Equal(AgentDescription, agentRecord.Value.Versions.Latest.Description); + Assert.Equal(AgentInstructions, definition.Instructions); + } + finally + { + // Cleanup. + await this._client.Agents.DeleteAgentAsync(agent.Name); + } + } + + [Theory(Skip = "For manual testing only")] + [InlineData("CreateWithChatClientAgentOptionsAsync")] + [InlineData("CreateWithChatClientAgentOptionsSync")] + [InlineData("CreateWithFoundryOptionsAsync")] + [InlineData("CreateWithFoundryOptionsSync")] + public async Task CreateAgent_CreatesAgentWithVectorStoresAsync(string createMechanism) + { + // Arrange. + string AgentName = AIProjectClientFixture.GenerateUniqueAgentName("VectorStoreAgent"); + const string AgentInstructions = """ + You are a helpful agent that can help fetch data from files you know about. + Use the File Search Tool to look up codes for words. + Do not answer a question unless you can find the answer using the File Search Tool. + """; + + // Get the project OpenAI client. + var projectOpenAIClient = this._client.GetProjectOpenAIClient(); + + // Create a vector store. + var searchFilePath = Path.GetTempFileName() + "wordcodelookup.txt"; + File.WriteAllText( + path: searchFilePath, + contents: "The word 'apple' uses the code 442345, while the word 'banana' uses the code 673457." + ); + OpenAIFile uploadedAgentFile = projectOpenAIClient.GetProjectFilesClient().UploadFile( + filePath: searchFilePath, + purpose: FileUploadPurpose.Assistants + ); + var vectorStoreMetadata = await projectOpenAIClient.GetProjectVectorStoresClient().CreateVectorStoreAsync(options: new() { FileIds = { uploadedAgentFile.Id }, Name = "WordCodeLookup_VectorStore" }); + + // Act. + var agent = createMechanism switch + { + "CreateWithChatClientAgentOptionsAsync" => await this._client.CreateAIAgentAsync( + model: s_config.DeploymentName, + name: AgentName, + instructions: AgentInstructions, + tools: [new HostedFileSearchTool() { Inputs = [new HostedVectorStoreContent(vectorStoreMetadata.Value.Id)] }]), + "CreateWithChatClientAgentOptionsSync" => this._client.CreateAIAgent( + model: s_config.DeploymentName, + name: AgentName, + instructions: AgentInstructions, + tools: [new HostedFileSearchTool() { Inputs = [new HostedVectorStoreContent(vectorStoreMetadata.Value.Id)] }]), + "CreateWithFoundryOptionsAsync" => await this._client.CreateAIAgentAsync( + model: s_config.DeploymentName, + name: AgentName, + instructions: AgentInstructions, + tools: [ResponseTool.CreateFileSearchTool(vectorStoreIds: [vectorStoreMetadata.Value.Id]).AsAITool()]), + "CreateWithFoundryOptionsSync" => this._client.CreateAIAgent( + model: s_config.DeploymentName, + name: AgentName, + instructions: AgentInstructions, + tools: [ResponseTool.CreateFileSearchTool(vectorStoreIds: [vectorStoreMetadata.Value.Id]).AsAITool()]), + _ => throw new InvalidOperationException($"Unknown create mechanism: {createMechanism}") + }; + + try + { + // Assert. + // Verify that the agent can use the vector store to answer a question. + var result = await agent.RunAsync("Can you give me the documented code for 'banana'?"); + Assert.Contains("673457", result.ToString()); + } + finally + { + // Cleanup. + await this._client.Agents.DeleteAgentAsync(agent.Name); + await projectOpenAIClient.GetProjectVectorStoresClient().DeleteVectorStoreAsync(vectorStoreMetadata.Value.Id); + await projectOpenAIClient.GetProjectFilesClient().DeleteFileAsync(uploadedAgentFile.Id); + File.Delete(searchFilePath); + } + } + + [Theory] + [InlineData("CreateWithChatClientAgentOptionsAsync")] + [InlineData("CreateWithChatClientAgentOptionsSync")] + [InlineData("CreateWithFoundryOptionsAsync")] + [InlineData("CreateWithFoundryOptionsSync")] + public async Task CreateAgent_CreatesAgentWithCodeInterpreterAsync(string createMechanism) + { + // Arrange. + string AgentName = AIProjectClientFixture.GenerateUniqueAgentName("CodeInterpreterAgent"); + const string AgentInstructions = """ + You are a helpful coding agent. A Python file is provided. Use the Code Interpreter Tool to run the file + and report the SECRET_NUMBER value it prints. Respond only with the number. + """; + + // Get the project OpenAI client. + var projectOpenAIClient = this._client.GetProjectOpenAIClient(); + + // Create a python file that prints a known value. + var codeFilePath = Path.GetTempFileName() + "secret_number.py"; + File.WriteAllText( + path: codeFilePath, + contents: "print(\"SECRET_NUMBER=24601\")" // Deterministic output we will look for. + ); + OpenAIFile uploadedCodeFile = projectOpenAIClient.GetProjectFilesClient().UploadFile( + filePath: codeFilePath, + purpose: FileUploadPurpose.Assistants + ); + + // Act. + var agent = createMechanism switch + { + // Hosted tool path (tools supplied via ChatClientAgentOptions) + "CreateWithChatClientAgentOptionsAsync" => await this._client.CreateAIAgentAsync( + model: s_config.DeploymentName, + name: AgentName, + instructions: AgentInstructions, + tools: [new HostedCodeInterpreterTool() { Inputs = [new HostedFileContent(uploadedCodeFile.Id)] }]), + "CreateWithChatClientAgentOptionsSync" => this._client.CreateAIAgent( + model: s_config.DeploymentName, + name: AgentName, + instructions: AgentInstructions, + tools: [new HostedCodeInterpreterTool() { Inputs = [new HostedFileContent(uploadedCodeFile.Id)] }]), + // Foundry (definitions + resources provided directly) + "CreateWithFoundryOptionsAsync" => await this._client.CreateAIAgentAsync( + model: s_config.DeploymentName, + name: AgentName, + instructions: AgentInstructions, + tools: [ResponseTool.CreateCodeInterpreterTool(new CodeInterpreterToolContainer(CodeInterpreterToolContainerConfiguration.CreateAutomaticContainerConfiguration([uploadedCodeFile.Id]))).AsAITool()]), + "CreateWithFoundryOptionsSync" => this._client.CreateAIAgent( + model: s_config.DeploymentName, + name: AgentName, + instructions: AgentInstructions, + tools: [ResponseTool.CreateCodeInterpreterTool(new CodeInterpreterToolContainer(CodeInterpreterToolContainerConfiguration.CreateAutomaticContainerConfiguration([uploadedCodeFile.Id]))).AsAITool()]), + _ => throw new InvalidOperationException($"Unknown create mechanism: {createMechanism}") + }; + + try + { + // Assert. + var result = await agent.RunAsync("What is the SECRET_NUMBER?"); + // We expect the model to run the code and surface the number. + Assert.Contains("24601", result.ToString()); + } + finally + { + // Cleanup. + await this._client.Agents.DeleteAgentAsync(agent.Name); + await projectOpenAIClient.GetProjectFilesClient().DeleteFileAsync(uploadedCodeFile.Id); + File.Delete(codeFilePath); + } + } + + [Theory] + [InlineData("CreateWithChatClientAgentOptionsAsync")] + [InlineData("CreateWithChatClientAgentOptionsSync")] + public async Task CreateAgent_CreatesAgentWithAIFunctionToolsAsync(string createMechanism) + { + // Arrange. + string AgentName = AIProjectClientFixture.GenerateUniqueAgentName("WeatherAgent"); + const string AgentInstructions = "You are a helpful weather assistant. Always call the GetWeather function to answer questions about weather."; + + static string GetWeather(string location) => $"The weather in {location} is sunny with a high of 23C."; + var weatherFunction = AIFunctionFactory.Create(GetWeather); + + ChatClientAgent agent = createMechanism switch + { + "CreateWithChatClientAgentOptionsAsync" => await this._client.CreateAIAgentAsync( + model: s_config.DeploymentName, + options: new ChatClientAgentOptions() + { + Name = AgentName, + ChatOptions = new() { Instructions = AgentInstructions, Tools = [weatherFunction] } + }), + "CreateWithChatClientAgentOptionsSync" => this._client.CreateAIAgent( + s_config.DeploymentName, + options: new ChatClientAgentOptions() + { + Name = AgentName, + ChatOptions = new() { Instructions = AgentInstructions, Tools = [weatherFunction] } + }), + _ => throw new InvalidOperationException($"Unknown create mechanism: {createMechanism}") + }; + + try + { + // Act. + var response = await agent.RunAsync("What is the weather like in Amsterdam?"); + + // Assert - ensure function was invoked and its output surfaced. + var text = response.Text; + Assert.Contains("Amsterdam", text, StringComparison.OrdinalIgnoreCase); + Assert.Contains("sunny", text, StringComparison.OrdinalIgnoreCase); + Assert.Contains("23", text, StringComparison.OrdinalIgnoreCase); + } + finally + { + await this._client.Agents.DeleteAgentAsync(agent.Name); + } + } +} diff --git a/dotnet/tests/AzureAI.IntegrationTests/AIProjectClientFixture.cs b/dotnet/tests/AzureAI.IntegrationTests/AIProjectClientFixture.cs new file mode 100644 index 0000000000..e982c8081f --- /dev/null +++ b/dotnet/tests/AzureAI.IntegrationTests/AIProjectClientFixture.cs @@ -0,0 +1,162 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Collections.Generic; +using System.Linq; +using System.Threading.Tasks; +using AgentConformance.IntegrationTests; +using AgentConformance.IntegrationTests.Support; +using Azure.AI.Projects; +using Azure.AI.Projects.OpenAI; +using Azure.Identity; +using Microsoft.Agents.AI; +using Microsoft.Extensions.AI; +using OpenAI.Responses; +using Shared.IntegrationTests; + +namespace AzureAI.IntegrationTests; + +public class AIProjectClientFixture : IChatClientAgentFixture +{ + private static readonly AzureAIConfiguration s_config = TestConfiguration.LoadSection(); + + private ChatClientAgent _agent = null!; + private AIProjectClient _client = null!; + + public IChatClient ChatClient => this._agent.ChatClient; + + public AIAgent Agent => this._agent; + + public async Task CreateConversationAsync() + { + var response = await this._client.GetProjectOpenAIClient().GetProjectConversationsClient().CreateProjectConversationAsync(); + return response.Value.Id; + } + + public async Task> GetChatHistoryAsync(AgentThread thread) + { + var chatClientThread = (ChatClientAgentThread)thread; + + if (chatClientThread.ConversationId?.StartsWith("conv_", StringComparison.OrdinalIgnoreCase) == true) + { + // Conversation threads do not persist message history. + return await this.GetChatHistoryFromConversationAsync(chatClientThread.ConversationId); + } + + if (chatClientThread.ConversationId?.StartsWith("resp_", StringComparison.OrdinalIgnoreCase) == true) + { + return await this.GetChatHistoryFromResponsesChainAsync(chatClientThread.ConversationId); + } + + return chatClientThread.MessageStore is null ? [] : (await chatClientThread.MessageStore.GetMessagesAsync()).ToList(); + } + + private async Task> GetChatHistoryFromResponsesChainAsync(string conversationId) + { + var openAIResponseClient = this._client.GetProjectOpenAIClient().GetProjectResponsesClient(); + var inputItems = await openAIResponseClient.GetResponseInputItemsAsync(conversationId).ToListAsync(); + var response = await openAIResponseClient.GetResponseAsync(conversationId); + var responseItem = response.Value.OutputItems.FirstOrDefault()!; + + // Take the messages that were the chat history leading up to the current response + // remove the instruction messages, and reverse the order so that the most recent message is last. + var previousMessages = inputItems + .Select(ConvertToChatMessage) + .Where(x => x.Text != "You are a helpful assistant.") + .Reverse(); + + // Convert the response item to a chat message. + var responseMessage = ConvertToChatMessage(responseItem); + + // Concatenate the previous messages with the response message to get a full chat history + // that includes the current response. + return [.. previousMessages, responseMessage]; + } + + private static ChatMessage ConvertToChatMessage(ResponseItem item) + { + if (item is MessageResponseItem messageResponseItem) + { + var role = messageResponseItem.Role == MessageRole.User ? ChatRole.User : ChatRole.Assistant; + return new ChatMessage(role, messageResponseItem.Content.FirstOrDefault()?.Text); + } + + throw new NotSupportedException("This test currently only supports text messages"); + } + + private async Task> GetChatHistoryFromConversationAsync(string conversationId) + { + List messages = []; + await foreach (AgentResponseItem item in this._client.GetProjectOpenAIClient().GetProjectConversationsClient().GetProjectConversationItemsAsync(conversationId, order: "asc")) + { + var openAIItem = item.AsOpenAIResponseItem(); + if (openAIItem is MessageResponseItem messageItem) + { + messages.Add(new ChatMessage + { + Role = new ChatRole(messageItem.Role.ToString()), + Contents = messageItem.Content + .Where(c => c.Kind is ResponseContentPartKind.OutputText or ResponseContentPartKind.InputText) + .Select(c => new TextContent(c.Text)) + .ToList() + }); + } + } + + return messages; + } + + public async Task CreateChatClientAgentAsync( + string name = "HelpfulAssistant", + string instructions = "You are a helpful assistant.", + IList? aiTools = null) + { + return await this._client.CreateAIAgentAsync(GenerateUniqueAgentName(name), model: s_config.DeploymentName, instructions: instructions, tools: aiTools); + } + + public static string GenerateUniqueAgentName(string baseName) => + $"{baseName}-{Guid.NewGuid().ToString("N").Substring(0, 8)}"; + + public Task DeleteAgentAsync(ChatClientAgent agent) => + this._client.Agents.DeleteAgentAsync(agent.Name); + + public async Task DeleteThreadAsync(AgentThread thread) + { + var typedThread = (ChatClientAgentThread)thread; + if (typedThread.ConversationId?.StartsWith("conv_", StringComparison.OrdinalIgnoreCase) == true) + { + await this._client.GetProjectOpenAIClient().GetProjectConversationsClient().DeleteConversationAsync(typedThread.ConversationId); + } + else if (typedThread.ConversationId?.StartsWith("resp_", StringComparison.OrdinalIgnoreCase) == true) + { + await this.DeleteResponseChainAsync(typedThread.ConversationId!); + } + } + + private async Task DeleteResponseChainAsync(string lastResponseId) + { + var response = await this._client.GetProjectOpenAIClient().GetProjectResponsesClient().GetResponseAsync(lastResponseId); + await this._client.GetProjectOpenAIClient().GetProjectResponsesClient().DeleteResponseAsync(lastResponseId); + + if (response.Value.PreviousResponseId is not null) + { + await this.DeleteResponseChainAsync(response.Value.PreviousResponseId); + } + } + + public Task DisposeAsync() + { + if (this._client is not null && this._agent is not null) + { + return this._client.Agents.DeleteAgentAsync(this._agent.Name); + } + + return Task.CompletedTask; + } + + public async Task InitializeAsync() + { + this._client = new(new Uri(s_config.Endpoint), new AzureCliCredential()); + this._agent = await this.CreateChatClientAgentAsync(); + } +} diff --git a/dotnet/tests/AzureAI.IntegrationTests/AzureAI.IntegrationTests.csproj b/dotnet/tests/AzureAI.IntegrationTests/AzureAI.IntegrationTests.csproj new file mode 100644 index 0000000000..83f65051d2 --- /dev/null +++ b/dotnet/tests/AzureAI.IntegrationTests/AzureAI.IntegrationTests.csproj @@ -0,0 +1,16 @@ + + + + True + + + + + + + + + + + + diff --git a/dotnet/tests/AzureAIAgentsPersistent.IntegrationTests/AzureAIAgentsPersistent.IntegrationTests.csproj b/dotnet/tests/AzureAIAgentsPersistent.IntegrationTests/AzureAIAgentsPersistent.IntegrationTests.csproj index 966ea64020..4078342410 100644 --- a/dotnet/tests/AzureAIAgentsPersistent.IntegrationTests/AzureAIAgentsPersistent.IntegrationTests.csproj +++ b/dotnet/tests/AzureAIAgentsPersistent.IntegrationTests/AzureAIAgentsPersistent.IntegrationTests.csproj @@ -1,8 +1,6 @@ - $(ProjectsTargetFrameworks) - $(ProjectsDebugTargetFrameworks) True diff --git a/dotnet/tests/AzureAIAgentsPersistent.IntegrationTests/AzureAIAgentsPersistentCreateTests.cs b/dotnet/tests/AzureAIAgentsPersistent.IntegrationTests/AzureAIAgentsPersistentCreateTests.cs index e3e9969a43..a10cc11d79 100644 --- a/dotnet/tests/AzureAIAgentsPersistent.IntegrationTests/AzureAIAgentsPersistentCreateTests.cs +++ b/dotnet/tests/AzureAIAgentsPersistent.IntegrationTests/AzureAIAgentsPersistentCreateTests.cs @@ -1,6 +1,7 @@ // Copyright (c) Microsoft. All rights reserved. using System; +using System.Diagnostics; using System.IO; using System.Threading.Tasks; using AgentConformance.IntegrationTests.Support; @@ -34,16 +35,20 @@ public async Task CreateAgent_CreatesAgentWithCorrectMetadataAsync(string create { "CreateWithChatClientAgentOptionsAsync" => await this._persistentAgentsClient.CreateAIAgentAsync( s_config.DeploymentName, - options: new ChatClientAgentOptions( - instructions: AgentInstructions, - name: AgentName, - description: AgentDescription)), + options: new ChatClientAgentOptions() + { + ChatOptions = new() { Instructions = AgentInstructions }, + Name = AgentName, + Description = AgentDescription + }), "CreateWithChatClientAgentOptionsSync" => this._persistentAgentsClient.CreateAIAgent( s_config.DeploymentName, - options: new ChatClientAgentOptions( - instructions: AgentInstructions, - name: AgentName, - description: AgentDescription)), + options: new ChatClientAgentOptions() + { + ChatOptions = new() { Instructions = AgentInstructions }, + Name = AgentName, + Description = AgentDescription + }), "CreateWithFoundryOptionsAsync" => await this._persistentAgentsClient.CreateAIAgentAsync( s_config.DeploymentName, instructions: AgentInstructions, @@ -104,19 +109,32 @@ You are a helpful agent that can help fetch data from files you know about. ); var vectorStoreMetadata = await this._persistentAgentsClient.VectorStores.CreateVectorStoreAsync([uploadedAgentFile.Id], name: "WordCodeLookup_VectorStore"); + // Wait for vector store indexing to complete before using it + await this.WaitForVectorStoreReadyAsync(this._persistentAgentsClient, vectorStoreMetadata.Value.Id); + // Act. var agent = createMechanism switch { "CreateWithChatClientAgentOptionsAsync" => await this._persistentAgentsClient.CreateAIAgentAsync( s_config.DeploymentName, - options: new ChatClientAgentOptions( - instructions: AgentInstructions, - tools: [new HostedFileSearchTool() { Inputs = [new HostedVectorStoreContent(vectorStoreMetadata.Value.Id)] }])), + options: new ChatClientAgentOptions() + { + ChatOptions = new() + { + Instructions = AgentInstructions, + Tools = [new HostedFileSearchTool() { Inputs = [new HostedVectorStoreContent(vectorStoreMetadata.Value.Id)] }] + } + }), "CreateWithChatClientAgentOptionsSync" => this._persistentAgentsClient.CreateAIAgent( s_config.DeploymentName, - options: new ChatClientAgentOptions( - instructions: AgentInstructions, - tools: [new HostedFileSearchTool() { Inputs = [new HostedVectorStoreContent(vectorStoreMetadata.Value.Id)] }])), + options: new ChatClientAgentOptions() + { + ChatOptions = new() + { + Instructions = AgentInstructions, + Tools = [new HostedFileSearchTool() { Inputs = [new HostedVectorStoreContent(vectorStoreMetadata.Value.Id)] }] + } + }), "CreateWithFoundryOptionsAsync" => await this._persistentAgentsClient.CreateAIAgentAsync( s_config.DeploymentName, instructions: AgentInstructions, @@ -179,15 +197,24 @@ and report the SECRET_NUMBER value it prints. Respond only with the number. // Hosted tool path (tools supplied via ChatClientAgentOptions) "CreateWithChatClientAgentOptionsAsync" => await this._persistentAgentsClient.CreateAIAgentAsync( s_config.DeploymentName, - options: new ChatClientAgentOptions( - instructions: AgentInstructions, - tools: [new HostedCodeInterpreterTool() { Inputs = [new HostedFileContent(uploadedCodeFile.Id)] }])), + options: new ChatClientAgentOptions() + { + ChatOptions = new() + { + Instructions = AgentInstructions, + Tools = [new HostedCodeInterpreterTool() { Inputs = [new HostedFileContent(uploadedCodeFile.Id)] }] + } + }), "CreateWithChatClientAgentOptionsSync" => this._persistentAgentsClient.CreateAIAgent( s_config.DeploymentName, - options: new ChatClientAgentOptions( - instructions: AgentInstructions, - tools: [new HostedCodeInterpreterTool() { Inputs = [new HostedFileContent(uploadedCodeFile.Id)] }])), - // Foundry (definitions + resources provided directly) + options: new ChatClientAgentOptions() + { + ChatOptions = new() + { + Instructions = AgentInstructions, + Tools = [new HostedCodeInterpreterTool() { Inputs = [new HostedFileContent(uploadedCodeFile.Id)] }] + } + }), "CreateWithFoundryOptionsAsync" => await this._persistentAgentsClient.CreateAIAgentAsync( s_config.DeploymentName, instructions: AgentInstructions, @@ -232,14 +259,24 @@ public async Task CreateAgent_CreatesAgentWithAIFunctionToolsAsync(string create { "CreateWithChatClientAgentOptionsAsync" => await this._persistentAgentsClient.CreateAIAgentAsync( s_config.DeploymentName, - options: new ChatClientAgentOptions( - instructions: AgentInstructions, - tools: [weatherFunction])), + options: new ChatClientAgentOptions() + { + ChatOptions = new() + { + Instructions = AgentInstructions, + Tools = [weatherFunction] + } + }), "CreateWithChatClientAgentOptionsSync" => this._persistentAgentsClient.CreateAIAgent( s_config.DeploymentName, - options: new ChatClientAgentOptions( - instructions: AgentInstructions, - tools: [weatherFunction])), + options: new ChatClientAgentOptions() + { + ChatOptions = new() + { + Instructions = AgentInstructions, + Tools = [weatherFunction] + } + }), _ => throw new InvalidOperationException($"Unknown create mechanism: {createMechanism}") }; @@ -259,4 +296,42 @@ public async Task CreateAgent_CreatesAgentWithAIFunctionToolsAsync(string create await this._persistentAgentsClient.Administration.DeleteAgentAsync(agent.Id); } } + + /// + /// Waits for a vector store to complete indexing by polling its status. + /// + /// The persistent agents client. + /// The ID of the vector store. + /// Maximum time to wait in seconds (default: 30). + /// A task that completes when the vector store is ready or throws on timeout/failure. + private async Task WaitForVectorStoreReadyAsync( + PersistentAgentsClient client, + string vectorStoreId, + int maxWaitSeconds = 30) + { + Stopwatch sw = Stopwatch.StartNew(); + while (sw.Elapsed.TotalSeconds < maxWaitSeconds) + { + PersistentAgentsVectorStore vectorStore = await client.VectorStores.GetVectorStoreAsync(vectorStoreId); + + if (vectorStore.Status == VectorStoreStatus.Completed) + { + if (vectorStore.FileCounts.Failed > 0) + { + throw new InvalidOperationException("Vector store indexing failed for some files"); + } + + return; + } + + if (vectorStore.Status == VectorStoreStatus.Expired) + { + throw new InvalidOperationException("Vector store has expired"); + } + + await Task.Delay(1000); + } + + throw new TimeoutException($"Vector store did not complete indexing within {maxWaitSeconds}s"); + } } diff --git a/dotnet/tests/CopilotStudio.IntegrationTests/CopilotStudio.IntegrationTests.csproj b/dotnet/tests/CopilotStudio.IntegrationTests/CopilotStudio.IntegrationTests.csproj index afbcc54f01..5f535eb7bd 100644 --- a/dotnet/tests/CopilotStudio.IntegrationTests/CopilotStudio.IntegrationTests.csproj +++ b/dotnet/tests/CopilotStudio.IntegrationTests/CopilotStudio.IntegrationTests.csproj @@ -1,8 +1,6 @@ - $(ProjectsTargetFrameworks) - $(ProjectsDebugTargetFrameworks) True true diff --git a/dotnet/tests/Directory.Build.props b/dotnet/tests/Directory.Build.props index 6c5a318e86..e6c285595e 100644 --- a/dotnet/tests/Directory.Build.props +++ b/dotnet/tests/Directory.Build.props @@ -6,7 +6,7 @@ false true false - net472;net9.0 + net10.0;net472 b7762d10-e29b-4bb1-8b74-b6d69a667dd4 $(NoWarn);Moq1410;xUnit2023 diff --git a/dotnet/tests/Microsoft.Agents.AI.A2A.UnitTests/A2AAgentTests.cs b/dotnet/tests/Microsoft.Agents.AI.A2A.UnitTests/A2AAgentTests.cs index 9399d99528..9869d47f6b 100644 --- a/dotnet/tests/Microsoft.Agents.AI.A2A.UnitTests/A2AAgentTests.cs +++ b/dotnet/tests/Microsoft.Agents.AI.A2A.UnitTests/A2AAgentTests.cs @@ -367,6 +367,7 @@ public async Task RunStreamingAsync_AllowsNonUserRoleMessagesAsync() // Act & Assert await foreach (var _ in this._agent.RunStreamingAsync(inputMessages)) { + // Just iterate through to trigger the logic } } @@ -396,15 +397,490 @@ public async Task RunAsync_WithHostedFileContent_ConvertsToFilePartAsync() Assert.Equal("https://example.com/file.pdf", ((FilePart)message.Parts[1]).File.Uri?.ToString()); } + [Fact] + public async Task RunAsync_WithContinuationTokenAndMessages_ThrowsInvalidOperationExceptionAsync() + { + // Arrange + var inputMessages = new List + { + new(ChatRole.User, "Test message") + }; + + var options = new AgentRunOptions { ContinuationToken = new A2AContinuationToken("task-123") }; + + // Act & Assert + await Assert.ThrowsAsync(() => this._agent.RunAsync(inputMessages, null, options)); + } + + [Fact] + public async Task RunAsync_WithContinuationToken_CallsGetTaskAsyncAsync() + { + // Arrange + this._handler.ResponseToReturn = new AgentTask + { + Id = "task-123", + ContextId = "context-123" + }; + + var options = new AgentRunOptions { ContinuationToken = new A2AContinuationToken("task-123") }; + + // Act + await this._agent.RunAsync([], options: options); + + // Assert + Assert.Equal("tasks/get", this._handler.CapturedJsonRpcRequest?.Method); + Assert.Equal("task-123", this._handler.CapturedTaskIdParams?.Id); + } + + [Fact] + public async Task RunAsync_WithTaskInThreadAndMessage_AddTaskAsReferencesToMessageAsync() + { + // Arrange + this._handler.ResponseToReturn = new AgentMessage + { + MessageId = "response-123", + Role = MessageRole.Agent, + Parts = [new TextPart { Text = "Response to task" }] + }; + + var thread = (A2AAgentThread)this._agent.GetNewThread(); + thread.TaskId = "task-123"; + + var inputMessage = new ChatMessage(ChatRole.User, "Please make the background transparent"); + + // Act + await this._agent.RunAsync(inputMessage, thread); + + // Assert + var message = this._handler.CapturedMessageSendParams?.Message; + Assert.Null(message?.TaskId); + Assert.NotNull(message?.ReferenceTaskIds); + Assert.Contains("task-123", message.ReferenceTaskIds); + } + + [Fact] + public async Task RunAsync_WithAgentTask_UpdatesThreadTaskIdAsync() + { + // Arrange + this._handler.ResponseToReturn = new AgentTask + { + Id = "task-456", + ContextId = "context-789", + Status = new() { State = TaskState.Submitted } + }; + + var thread = this._agent.GetNewThread(); + + // Act + await this._agent.RunAsync("Start a task", thread); + + // Assert + var a2aThread = (A2AAgentThread)thread; + Assert.Equal("task-456", a2aThread.TaskId); + } + + [Fact] + public async Task RunAsync_WithAgentTaskResponse_ReturnsTaskResponseCorrectlyAsync() + { + // Arrange + this._handler.ResponseToReturn = new AgentTask + { + Id = "task-789", + ContextId = "context-456", + Status = new() { State = TaskState.Submitted }, + Metadata = new Dictionary + { + { "key1", JsonSerializer.SerializeToElement("value1") }, + { "count", JsonSerializer.SerializeToElement(42) } + } + }; + + var thread = this._agent.GetNewThread(); + + // Act + var result = await this._agent.RunAsync("Start a long-running task", thread); + + // Assert - verify task is converted correctly + Assert.NotNull(result); + Assert.Equal(this._agent.Id, result.AgentId); + Assert.Equal("task-789", result.ResponseId); + + Assert.NotNull(result.RawRepresentation); + Assert.IsType(result.RawRepresentation); + Assert.Equal("task-789", ((AgentTask)result.RawRepresentation).Id); + + // Assert - verify continuation token is set for submitted task + Assert.NotNull(result.ContinuationToken); + Assert.IsType(result.ContinuationToken); + Assert.Equal("task-789", ((A2AContinuationToken)result.ContinuationToken).TaskId); + + // Assert - verify thread is updated with context and task IDs + var a2aThread = (A2AAgentThread)thread; + Assert.Equal("context-456", a2aThread.ContextId); + Assert.Equal("task-789", a2aThread.TaskId); + + // Assert - verify metadata is preserved + Assert.NotNull(result.AdditionalProperties); + Assert.NotNull(result.AdditionalProperties["key1"]); + Assert.Equal("value1", ((JsonElement)result.AdditionalProperties["key1"]!).GetString()); + Assert.NotNull(result.AdditionalProperties["count"]); + Assert.Equal(42, ((JsonElement)result.AdditionalProperties["count"]!).GetInt32()); + } + + [Theory] + [InlineData(TaskState.Submitted)] + [InlineData(TaskState.Working)] + [InlineData(TaskState.Completed)] + [InlineData(TaskState.Failed)] + [InlineData(TaskState.Canceled)] + public async Task RunAsync_WithVariousTaskStates_ReturnsCorrectTokenAsync(TaskState taskState) + { + // Arrange + this._handler.ResponseToReturn = new AgentTask + { + Id = "task-123", + ContextId = "context-123", + Status = new() { State = taskState } + }; + + // Act + var result = await this._agent.RunAsync("Test message"); + + // Assert + if (taskState is TaskState.Submitted or TaskState.Working) + { + Assert.NotNull(result.ContinuationToken); + } + else + { + Assert.Null(result.ContinuationToken); + } + } + + [Fact] + public async Task RunStreamingAsync_WithContinuationTokenAndMessages_ThrowsInvalidOperationExceptionAsync() + { + // Arrange + var inputMessages = new List + { + new(ChatRole.User, "Test message") + }; + + var options = new AgentRunOptions { ContinuationToken = new A2AContinuationToken("task-123") }; + + // Act & Assert + await Assert.ThrowsAsync(async () => + { + await foreach (var _ in this._agent.RunStreamingAsync(inputMessages, null, options)) + { + // Just iterate through to trigger the exception + } + }); + } + + [Fact] + public async Task RunStreamingAsync_WithTaskInThreadAndMessage_AddTaskAsReferencesToMessageAsync() + { + // Arrange + this._handler.StreamingResponseToReturn = new AgentMessage + { + MessageId = "response-123", + Role = MessageRole.Agent, + Parts = [new TextPart { Text = "Response to task" }] + }; + + var thread = (A2AAgentThread)this._agent.GetNewThread(); + thread.TaskId = "task-123"; + + // Act + await foreach (var _ in this._agent.RunStreamingAsync("Please make the background transparent", thread)) + { + // Just iterate through to trigger the logic + } + + // Assert + var message = this._handler.CapturedMessageSendParams?.Message; + Assert.Null(message?.TaskId); + Assert.NotNull(message?.ReferenceTaskIds); + Assert.Contains("task-123", message.ReferenceTaskIds); + } + + [Fact] + public async Task RunStreamingAsync_WithAgentTask_UpdatesThreadTaskIdAsync() + { + // Arrange + this._handler.StreamingResponseToReturn = new AgentTask + { + Id = "task-456", + ContextId = "context-789", + Status = new() { State = TaskState.Submitted } + }; + + var thread = this._agent.GetNewThread(); + + // Act + await foreach (var _ in this._agent.RunStreamingAsync("Start a task", thread)) + { + // Just iterate through to trigger the logic + } + + // Assert + var a2aThread = (A2AAgentThread)thread; + Assert.Equal("task-456", a2aThread.TaskId); + } + + [Fact] + public async Task RunStreamingAsync_WithAgentMessage_YieldsResponseUpdateAsync() + { + // Arrange + const string MessageId = "msg-123"; + const string ContextId = "ctx-456"; + const string MessageText = "Hello from agent!"; + + this._handler.StreamingResponseToReturn = new AgentMessage + { + MessageId = MessageId, + Role = MessageRole.Agent, + ContextId = ContextId, + Parts = + [ + new TextPart { Text = MessageText } + ] + }; + + // Act + var updates = new List(); + await foreach (var update in this._agent.RunStreamingAsync("Test message")) + { + updates.Add(update); + } + + // Assert - one update should be yielded + Assert.Single(updates); + + var update0 = updates[0]; + Assert.Equal(ChatRole.Assistant, update0.Role); + Assert.Equal(MessageId, update0.MessageId); + Assert.Equal(MessageId, update0.ResponseId); + Assert.Equal(this._agent.Id, update0.AgentId); + Assert.Equal(MessageText, update0.Text); + Assert.IsType(update0.RawRepresentation); + Assert.Equal(MessageId, ((AgentMessage)update0.RawRepresentation!).MessageId); + } + + [Fact] + public async Task RunStreamingAsync_WithAgentTask_YieldsResponseUpdateAsync() + { + // Arrange + const string TaskId = "task-789"; + const string ContextId = "ctx-012"; + + this._handler.StreamingResponseToReturn = new AgentTask + { + Id = TaskId, + ContextId = ContextId, + Status = new() { State = TaskState.Submitted }, + Artifacts = [ + new() + { + ArtifactId = "art-123", + Parts = [new TextPart { Text = "Task artifact content" }] + } + ] + }; + + var thread = this._agent.GetNewThread(); + + // Act + var updates = new List(); + await foreach (var update in this._agent.RunStreamingAsync("Start long-running task", thread)) + { + updates.Add(update); + } + + // Assert - one update should be yielded from artifact + Assert.Single(updates); + + var update0 = updates[0]; + Assert.Equal(ChatRole.Assistant, update0.Role); + Assert.Equal(TaskId, update0.ResponseId); + Assert.Equal(this._agent.Id, update0.AgentId); + Assert.IsType(update0.RawRepresentation); + Assert.Equal(TaskId, ((AgentTask)update0.RawRepresentation!).Id); + + // Assert - thread should be updated with context and task IDs + var a2aThread = (A2AAgentThread)thread; + Assert.Equal(ContextId, a2aThread.ContextId); + Assert.Equal(TaskId, a2aThread.TaskId); + } + + [Fact] + public async Task RunStreamingAsync_WithTaskStatusUpdateEvent_YieldsResponseUpdateAsync() + { + // Arrange + const string TaskId = "task-status-123"; + const string ContextId = "ctx-status-456"; + + this._handler.StreamingResponseToReturn = new TaskStatusUpdateEvent + { + TaskId = TaskId, + ContextId = ContextId, + Status = new() { State = TaskState.Working } + }; + + var thread = this._agent.GetNewThread(); + + // Act + var updates = new List(); + await foreach (var update in this._agent.RunStreamingAsync("Check task status", thread)) + { + updates.Add(update); + } + + // Assert - one update should be yielded + Assert.Single(updates); + + var update0 = updates[0]; + Assert.Equal(ChatRole.Assistant, update0.Role); + Assert.Equal(TaskId, update0.ResponseId); + Assert.Equal(this._agent.Id, update0.AgentId); + Assert.IsType(update0.RawRepresentation); + + // Assert - thread should be updated with context and task IDs + var a2aThread = (A2AAgentThread)thread; + Assert.Equal(ContextId, a2aThread.ContextId); + Assert.Equal(TaskId, a2aThread.TaskId); + } + + [Fact] + public async Task RunStreamingAsync_WithTaskArtifactUpdateEvent_YieldsResponseUpdateAsync() + { + // Arrange + const string TaskId = "task-artifact-123"; + const string ContextId = "ctx-artifact-456"; + const string ArtifactContent = "Task artifact data"; + + this._handler.StreamingResponseToReturn = new TaskArtifactUpdateEvent + { + TaskId = TaskId, + ContextId = ContextId, + Artifact = new() + { + ArtifactId = "artifact-789", + Parts = [new TextPart { Text = ArtifactContent }] + } + }; + + var thread = this._agent.GetNewThread(); + + // Act + var updates = new List(); + await foreach (var update in this._agent.RunStreamingAsync("Process artifact", thread)) + { + updates.Add(update); + } + + // Assert - one update should be yielded + Assert.Single(updates); + + var update0 = updates[0]; + Assert.Equal(ChatRole.Assistant, update0.Role); + Assert.Equal(TaskId, update0.ResponseId); + Assert.Equal(this._agent.Id, update0.AgentId); + Assert.IsType(update0.RawRepresentation); + + // Assert - artifact content should be in the update + Assert.NotEmpty(update0.Contents); + Assert.Equal(ArtifactContent, update0.Text); + + // Assert - thread should be updated with context and task IDs + var a2aThread = (A2AAgentThread)thread; + Assert.Equal(ContextId, a2aThread.ContextId); + Assert.Equal(TaskId, a2aThread.TaskId); + } + + [Fact] + public async Task RunAsync_WithAllowBackgroundResponsesAndNoThread_ThrowsInvalidOperationExceptionAsync() + { + // Arrange + var inputMessages = new List + { + new(ChatRole.User, "Test message") + }; + + var options = new AgentRunOptions { AllowBackgroundResponses = true }; + + // Act & Assert + await Assert.ThrowsAsync(() => this._agent.RunAsync(inputMessages, null, options)); + } + + [Fact] + public async Task RunStreamingAsync_WithAllowBackgroundResponsesAndNoThread_ThrowsInvalidOperationExceptionAsync() + { + // Arrange + var inputMessages = new List + { + new(ChatRole.User, "Test message") + }; + + var options = new AgentRunOptions { AllowBackgroundResponses = true }; + + // Act & Assert + await Assert.ThrowsAsync(async () => + { + await foreach (var _ in this._agent.RunStreamingAsync(inputMessages, null, options)) + { + // Just iterate through to trigger the exception + } + }); + } + + [Fact] + public async Task RunAsync_WithInvalidThreadType_ThrowsInvalidOperationExceptionAsync() + { + // Arrange + // Create a thread from a different agent type + var invalidThread = new CustomAgentThread(); + + // Act & Assert + await Assert.ThrowsAsync(() => this._agent.RunAsync(invalidThread)); + } + + [Fact] + public async Task RunStreamingAsync_WithInvalidThreadType_ThrowsInvalidOperationExceptionAsync() + { + // Arrange + var inputMessages = new List + { + new(ChatRole.User, "Test message") + }; + + // Create a thread from a different agent type + var invalidThread = new CustomAgentThread(); + + // Act & Assert + await Assert.ThrowsAsync(async () => await this._agent.RunStreamingAsync(inputMessages, invalidThread).ToListAsync()); + } + public void Dispose() { this._handler.Dispose(); this._httpClient.Dispose(); } + + /// + /// Custom agent thread class for testing invalid thread type scenario. + /// + private sealed class CustomAgentThread : AgentThread; + internal sealed class A2AClientHttpMessageHandlerStub : HttpMessageHandler { + public JsonRpcRequest? CapturedJsonRpcRequest { get; set; } + public MessageSendParams? CapturedMessageSendParams { get; set; } + public TaskIdParams? CapturedTaskIdParams { get; set; } + public A2AEvent? ResponseToReturn { get; set; } public A2AEvent? StreamingResponseToReturn { get; set; } @@ -416,9 +892,19 @@ protected override async Task SendAsync(HttpRequestMessage var content = await request.Content!.ReadAsStringAsync(); #pragma warning restore CA2016 - var jsonRpcRequest = JsonSerializer.Deserialize(content)!; + this.CapturedJsonRpcRequest = JsonSerializer.Deserialize(content); - this.CapturedMessageSendParams = jsonRpcRequest.Params?.Deserialize(); + try + { + this.CapturedMessageSendParams = this.CapturedJsonRpcRequest?.Params?.Deserialize(); + } + catch { /* Ignore deserialization errors for non-MessageSendParams requests */ } + + try + { + this.CapturedTaskIdParams = this.CapturedJsonRpcRequest?.Params?.Deserialize(); + } + catch { /* Ignore deserialization errors for non-TaskIdParams requests */ } // Return the pre-configured non-streaming response if (this.ResponseToReturn is not null) diff --git a/dotnet/tests/Microsoft.Agents.AI.A2A.UnitTests/A2AAgentThreadTests.cs b/dotnet/tests/Microsoft.Agents.AI.A2A.UnitTests/A2AAgentThreadTests.cs new file mode 100644 index 0000000000..90b65aa5ac --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.A2A.UnitTests/A2AAgentThreadTests.cs @@ -0,0 +1,30 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json; + +namespace Microsoft.Agents.AI.A2A.UnitTests; + +/// +/// Unit tests for the class. +/// +public sealed class A2AAgentThreadTests +{ + [Fact] + public void Constructor_RoundTrip_SerializationPreservesState() + { + // Arrange + const string ContextId = "context-rt-001"; + const string TaskId = "task-rt-002"; + + A2AAgentThread originalThread = new() { ContextId = ContextId, TaskId = TaskId }; + + // Act + JsonElement serialized = originalThread.Serialize(); + + A2AAgentThread deserializedThread = new(serialized); + + // Assert + Assert.Equal(originalThread.ContextId, deserializedThread.ContextId); + Assert.Equal(originalThread.TaskId, deserializedThread.TaskId); + } +} diff --git a/dotnet/tests/Microsoft.Agents.AI.A2A.UnitTests/A2AContinuationTokenTests.cs b/dotnet/tests/Microsoft.Agents.AI.A2A.UnitTests/A2AContinuationTokenTests.cs new file mode 100644 index 0000000000..1bb0d99e00 --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.A2A.UnitTests/A2AContinuationTokenTests.cs @@ -0,0 +1,152 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Text.Json; +using Microsoft.Extensions.AI; + +namespace Microsoft.Agents.AI.A2A.UnitTests; + +/// +/// Unit tests for the class. +/// +public sealed class A2AContinuationTokenTests +{ + [Fact] + public void Constructor_WithValidTaskId_InitializesTaskIdProperty() + { + // Arrange + const string TaskId = "task-123"; + + // Act + var token = new A2AContinuationToken(TaskId); + + // Assert + Assert.Equal(TaskId, token.TaskId); + } + + [Fact] + public void ToBytes_WithValidToken_SerializesToJsonBytes() + { + // Arrange + const string TaskId = "task-456"; + var token = new A2AContinuationToken(TaskId); + + // Act + var bytes = token.ToBytes(); + + // Assert + Assert.NotEqual(0, bytes.Length); + var jsonString = System.Text.Encoding.UTF8.GetString(bytes.ToArray()); + using var jsonDoc = JsonDocument.Parse(jsonString); + var root = jsonDoc.RootElement; + Assert.True(root.TryGetProperty("taskId", out var taskIdElement)); + Assert.Equal(TaskId, taskIdElement.GetString()); + } + + [Fact] + public void FromToken_WithA2AContinuationToken_ReturnsSameInstance() + { + // Arrange + const string TaskId = "task-direct"; + var originalToken = new A2AContinuationToken(TaskId); + + // Act + var resultToken = A2AContinuationToken.FromToken(originalToken); + + // Assert + Assert.Same(originalToken, resultToken); + Assert.Equal(TaskId, resultToken.TaskId); + } + + [Fact] + public void FromToken_WithSerializedToken_DeserializesCorrectly() + { + // Arrange + const string TaskId = "task-deserialized"; + var originalToken = new A2AContinuationToken(TaskId); + var serialized = originalToken.ToBytes(); + + // Create a mock token wrapper to pass to FromToken + var mockToken = new MockResponseContinuationToken(serialized); + + // Act + var resultToken = A2AContinuationToken.FromToken(mockToken); + + // Assert + Assert.Equal(TaskId, resultToken.TaskId); + Assert.IsType(resultToken); + } + + [Fact] + public void FromToken_RoundTrip_PreservesTaskId() + { + // Arrange + const string TaskId = "task-roundtrip-123"; + var originalToken = new A2AContinuationToken(TaskId); + var serialized = originalToken.ToBytes(); + var mockToken = new MockResponseContinuationToken(serialized); + + // Act + var deserializedToken = A2AContinuationToken.FromToken(mockToken); + var reserialized = deserializedToken.ToBytes(); + var mockToken2 = new MockResponseContinuationToken(reserialized); + var deserializedAgain = A2AContinuationToken.FromToken(mockToken2); + + // Assert + Assert.Equal(TaskId, deserializedAgain.TaskId); + } + + [Fact] + public void FromToken_WithEmptyData_ThrowsArgumentException() + { + // Arrange + var emptyToken = new MockResponseContinuationToken(ReadOnlyMemory.Empty); + + // Act & Assert + Assert.Throws(() => A2AContinuationToken.FromToken(emptyToken)); + } + + [Fact] + public void FromToken_WithMissingTaskIdProperty_ThrowsException() + { + // Arrange + var jsonWithoutTaskId = System.Text.Encoding.UTF8.GetBytes("{ \"someOtherProperty\": \"value\" }").AsMemory(); + var mockToken = new MockResponseContinuationToken(jsonWithoutTaskId); + + // Act & Assert + Assert.Throws(() => A2AContinuationToken.FromToken(mockToken)); + } + + [Fact] + public void FromToken_WithValidTaskId_ParsesTaskIdCorrectly() + { + // Arrange + const string TaskId = "task-multi-prop"; + var json = System.Text.Encoding.UTF8.GetBytes($"{{ \"taskId\": \"{TaskId}\" }}").AsMemory(); + var mockToken = new MockResponseContinuationToken(json); + + // Act + var resultToken = A2AContinuationToken.FromToken(mockToken); + + // Assert + Assert.Equal(TaskId, resultToken.TaskId); + } + + /// + /// Mock implementation of ResponseContinuationToken for testing. + /// + private sealed class MockResponseContinuationToken : ResponseContinuationToken + { + private readonly ReadOnlyMemory _data; + + public MockResponseContinuationToken(ReadOnlyMemory data) + { + this._data = data; + } + + public override ReadOnlyMemory ToBytes() + { + return this._data; + } + } +} diff --git a/dotnet/tests/Microsoft.Agents.AI.A2A.UnitTests/Extensions/A2AAgentTaskExtensionsTests.cs b/dotnet/tests/Microsoft.Agents.AI.A2A.UnitTests/Extensions/A2AAgentTaskExtensionsTests.cs new file mode 100644 index 0000000000..97c9ca7c05 --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.A2A.UnitTests/Extensions/A2AAgentTaskExtensionsTests.cs @@ -0,0 +1,169 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Collections.Generic; +using A2A; +using Microsoft.Extensions.AI; + +namespace Microsoft.Agents.AI.A2A.UnitTests; + +/// +/// Unit tests for the class. +/// +public sealed class A2AAgentTaskExtensionsTests +{ + [Fact] + public void ToChatMessages_WithNullAgentTask_ThrowsArgumentNullException() + { + // Arrange + AgentTask agentTask = null!; + + // Act & Assert + Assert.Throws(() => agentTask.ToChatMessages()); + } + + [Fact] + public void ToAIContents_WithNullAgentTask_ThrowsArgumentNullException() + { + // Arrange + AgentTask agentTask = null!; + + // Act & Assert + Assert.Throws(() => agentTask.ToAIContents()); + } + + [Fact] + public void ToChatMessages_WithEmptyArtifactsAndNoUserInputRequests_ReturnsNull() + { + // Arrange + var agentTask = new AgentTask + { + Id = "task1", + Artifacts = [], + Status = new AgentTaskStatus { State = TaskState.Completed }, + }; + + // Act + IList? result = agentTask.ToChatMessages(); + + // Assert + Assert.Null(result); + } + + [Fact] + public void ToChatMessages_WithNullArtifactsAndNoUserInputRequests_ReturnsNull() + { + // Arrange + var agentTask = new AgentTask + { + Id = "task1", + Artifacts = null, + Status = new AgentTaskStatus { State = TaskState.Completed }, + }; + + // Act + IList? result = agentTask.ToChatMessages(); + + // Assert + Assert.Null(result); + } + + [Fact] + public void ToAIContents_WithEmptyArtifactsAndNoUserInputRequests_ReturnsNull() + { + // Arrange + var agentTask = new AgentTask + { + Id = "task1", + Artifacts = [], + Status = new AgentTaskStatus { State = TaskState.Completed }, + }; + + // Act + IList? result = agentTask.ToAIContents(); + + // Assert + Assert.Null(result); + } + + [Fact] + public void ToAIContents_WithNullArtifactsAndNoUserInputRequests_ReturnsNull() + { + // Arrange + var agentTask = new AgentTask + { + Id = "task1", + Artifacts = null, + Status = new AgentTaskStatus { State = TaskState.Completed }, + }; + + // Act + IList? result = agentTask.ToAIContents(); + + // Assert + Assert.Null(result); + } + + [Fact] + public void ToChatMessages_WithValidArtifact_ReturnsChatMessages() + { + // Arrange + var artifact = new Artifact + { + Parts = [new TextPart { Text = "response" }], + }; + + var agentTask = new AgentTask + { + Id = "task1", + Artifacts = [artifact], + Status = new AgentTaskStatus { State = TaskState.Completed }, + }; + + // Act + IList? result = agentTask.ToChatMessages(); + + // Assert + Assert.NotNull(result); + Assert.NotEmpty(result); + Assert.All(result, msg => Assert.Equal(ChatRole.Assistant, msg.Role)); + Assert.Equal("response", result[0].Contents[0].ToString()); + } + + [Fact] + public void ToAIContents_WithMultipleArtifacts_FlattenAllContents() + { + // Arrange + var artifact1 = new Artifact + { + Parts = [new TextPart { Text = "content1" }], + }; + + var artifact2 = new Artifact + { + Parts = + [ + new TextPart { Text = "content2" }, + new TextPart { Text = "content3" } + ], + }; + + var agentTask = new AgentTask + { + Id = "task1", + Artifacts = [artifact1, artifact2], + Status = new AgentTaskStatus { State = TaskState.Completed }, + }; + + // Act + IList? result = agentTask.ToAIContents(); + + // Assert + Assert.NotNull(result); + Assert.NotEmpty(result); + Assert.Equal(3, result.Count); + Assert.Equal("content1", result[0].ToString()); + Assert.Equal("content2", result[1].ToString()); + Assert.Equal("content3", result[2].ToString()); + } +} diff --git a/dotnet/tests/Microsoft.Agents.AI.A2A.UnitTests/Extensions/A2AArtifactExtensionsTests.cs b/dotnet/tests/Microsoft.Agents.AI.A2A.UnitTests/Extensions/A2AArtifactExtensionsTests.cs new file mode 100644 index 0000000000..b18abd4485 --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.A2A.UnitTests/Extensions/A2AArtifactExtensionsTests.cs @@ -0,0 +1,107 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Collections.Generic; +using System.Text.Json; +using A2A; +using Microsoft.Extensions.AI; + +namespace Microsoft.Agents.AI.A2A.UnitTests; + +/// +/// Unit tests for the class. +/// +public sealed class A2AArtifactExtensionsTests +{ + [Fact] + public void ToChatMessage_WithMultiplePartsMetadataAndRawRepresentation_ReturnsCorrectChatMessage() + { + // Arrange + var artifact = new Artifact + { + ArtifactId = "artifact-comprehensive", + Name = "comprehensive-artifact", + Parts = + [ + new TextPart { Text = "First part" }, + new TextPart { Text = "Second part" }, + new TextPart { Text = "Third part" } + ], + Metadata = new Dictionary + { + { "key1", JsonSerializer.SerializeToElement("value1") }, + { "key2", JsonSerializer.SerializeToElement(42) } + } + }; + + // Act + var result = artifact.ToChatMessage(); + + // Assert - Verify multiple parts + Assert.NotNull(result); + Assert.Equal(ChatRole.Assistant, result.Role); + Assert.Equal(3, result.Contents.Count); + Assert.All(result.Contents, content => Assert.IsType(content)); + Assert.Equal("First part", ((TextContent)result.Contents[0]).Text); + Assert.Equal("Second part", ((TextContent)result.Contents[1]).Text); + Assert.Equal("Third part", ((TextContent)result.Contents[2]).Text); + + // Assert - Verify metadata conversion to AdditionalProperties + Assert.NotNull(result.AdditionalProperties); + Assert.Equal(2, result.AdditionalProperties.Count); + Assert.True(result.AdditionalProperties.ContainsKey("key1")); + Assert.True(result.AdditionalProperties.ContainsKey("key2")); + + // Assert - Verify RawRepresentation is set to artifact + Assert.NotNull(result.RawRepresentation); + Assert.Same(artifact, result.RawRepresentation); + } + + [Fact] + public void ToAIContents_WithMultipleParts_ReturnsCorrectList() + { + // Arrange + var artifact = new Artifact + { + ArtifactId = "artifact-ai-multi", + Name = "test", + Parts = + [ + new TextPart { Text = "Part 1" }, + new TextPart { Text = "Part 2" }, + new TextPart { Text = "Part 3" } + ], + Metadata = null + }; + + // Act + var result = artifact.ToAIContents(); + + // Assert + Assert.NotNull(result); + Assert.Equal(3, result.Count); + Assert.All(result, content => Assert.IsType(content)); + Assert.Equal("Part 1", ((TextContent)result[0]).Text); + Assert.Equal("Part 2", ((TextContent)result[1]).Text); + Assert.Equal("Part 3", ((TextContent)result[2]).Text); + } + + [Fact] + public void ToAIContents_WithEmptyParts_ReturnsEmptyList() + { + // Arrange + var artifact = new Artifact + { + ArtifactId = "artifact-empty", + Name = "test", + Parts = [], + Metadata = null + }; + + // Act + var result = artifact.ToAIContents(); + + // Assert + Assert.NotNull(result); + Assert.Empty(result); + } +} diff --git a/dotnet/tests/Microsoft.Agents.AI.A2A.UnitTests/Microsoft.Agents.AI.A2A.UnitTests.csproj b/dotnet/tests/Microsoft.Agents.AI.A2A.UnitTests/Microsoft.Agents.AI.A2A.UnitTests.csproj index f654f3eeec..d33de0613b 100644 --- a/dotnet/tests/Microsoft.Agents.AI.A2A.UnitTests/Microsoft.Agents.AI.A2A.UnitTests.csproj +++ b/dotnet/tests/Microsoft.Agents.AI.A2A.UnitTests/Microsoft.Agents.AI.A2A.UnitTests.csproj @@ -1,14 +1,5 @@ - - $(ProjectsTargetFrameworks) - - - - - - - diff --git a/dotnet/tests/Microsoft.Agents.AI.AGUI.UnitTests/AGUIAgentTests.cs b/dotnet/tests/Microsoft.Agents.AI.AGUI.UnitTests/AGUIAgentTests.cs deleted file mode 100644 index d6388ff711..0000000000 --- a/dotnet/tests/Microsoft.Agents.AI.AGUI.UnitTests/AGUIAgentTests.cs +++ /dev/null @@ -1,344 +0,0 @@ -// Copyright (c) Microsoft. All rights reserved. - -using System; -using System.Collections.Generic; -using System.Linq; -using System.Net; -using System.Net.Http; -using System.Text.Json; -using System.Threading; -using System.Threading.Tasks; -using Microsoft.Agents.AI.AGUI.Shared; -using Microsoft.Extensions.AI; -using Moq; -using Moq.Protected; - -namespace Microsoft.Agents.AI.AGUI.UnitTests; - -/// -/// Unit tests for the class. -/// -public sealed class AGUIAgentTests -{ - [Fact] - public async Task RunAsync_AggregatesStreamingUpdates_ReturnsCompleteMessagesAsync() - { - // Arrange - using HttpClient httpClient = this.CreateMockHttpClient(new BaseEvent[] - { - new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, - new TextMessageStartEvent { MessageId = "msg1", Role = AGUIRoles.Assistant }, - new TextMessageContentEvent { MessageId = "msg1", Delta = "Hello" }, - new TextMessageContentEvent { MessageId = "msg1", Delta = " World" }, - new TextMessageEndEvent { MessageId = "msg1" }, - new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } - }); - - AGUIAgent agent = new("agent1", "Test agent", httpClient, "http://localhost/agent"); - List messages = [new ChatMessage(ChatRole.User, "Test")]; - - // Act - AgentRunResponse response = await agent.RunAsync(messages); - - // Assert - Assert.NotNull(response); - Assert.NotEmpty(response.Messages); - ChatMessage message = response.Messages.First(); - Assert.Equal(ChatRole.Assistant, message.Role); - Assert.Equal("Hello World", message.Text); - } - - [Fact] - public async Task RunAsync_WithEmptyUpdateStream_ContainsOnlyMetadataMessagesAsync() - { - // Arrange - using HttpClient httpClient = this.CreateMockHttpClient( - [ - new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, - new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } - ]); - - AGUIAgent agent = new("agent1", "Test agent", httpClient, "http://localhost/agent"); - List messages = [new ChatMessage(ChatRole.User, "Test")]; - - // Act - AgentRunResponse response = await agent.RunAsync(messages); - - // Assert - Assert.NotNull(response); - // RunStarted and RunFinished events are aggregated into messages by ToChatResponse() - Assert.NotEmpty(response.Messages); - Assert.All(response.Messages, m => Assert.Equal(ChatRole.Assistant, m.Role)); - } - - [Fact] - public async Task RunAsync_WithNullMessages_ThrowsArgumentNullExceptionAsync() - { - // Arrange - using HttpClient httpClient = new(); - AGUIAgent agent = new("agent1", "Test agent", httpClient, "http://localhost/agent"); - - // Act & Assert - await Assert.ThrowsAsync(() => agent.RunAsync(messages: null!)); - } - - [Fact] - public async Task RunAsync_WithNullThread_CreatesNewThreadAsync() - { - // Arrange - using HttpClient httpClient = this.CreateMockHttpClient( - [ - new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, - new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } - ]); - - AGUIAgent agent = new("agent1", "Test agent", httpClient, "http://localhost/agent"); - List messages = [new ChatMessage(ChatRole.User, "Test")]; - - // Act - AgentRunResponse response = await agent.RunAsync(messages, thread: null); - - // Assert - Assert.NotNull(response); - } - - [Fact] - public async Task RunAsync_WithNonAGUIAgentThread_ThrowsInvalidOperationExceptionAsync() - { - // Arrange - using HttpClient httpClient = new(); - AGUIAgent agent = new("agent1", "Test agent", httpClient, "http://localhost/agent"); - List messages = [new ChatMessage(ChatRole.User, "Test")]; - AgentThread invalidThread = new TestInMemoryAgentThread(); - - // Act & Assert - await Assert.ThrowsAsync(() => agent.RunAsync(messages, thread: invalidThread)); - } - - [Fact] - public async Task RunStreamingAsync_YieldsAllEvents_FromServerStreamAsync() - { - // Arrange - using HttpClient httpClient = this.CreateMockHttpClient( - [ - new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, - new TextMessageStartEvent { MessageId = "msg1", Role = AGUIRoles.Assistant }, - new TextMessageContentEvent { MessageId = "msg1", Delta = "Hello" }, - new TextMessageEndEvent { MessageId = "msg1" }, - new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } - ]); - - AGUIAgent agent = new("agent1", "Test agent", httpClient, "http://localhost/agent"); - List messages = [new ChatMessage(ChatRole.User, "Test")]; - - // Act - List updates = []; - await foreach (AgentRunResponseUpdate update in agent.RunStreamingAsync(messages)) - { - // Consume the stream - updates.Add(update); - } - - // Assert - Assert.NotEmpty(updates); - Assert.Contains(updates, u => u.ResponseId != null); // RunStarted sets ResponseId - Assert.Contains(updates, u => u.Contents.Any(c => c is TextContent)); - Assert.Contains(updates, u => u.Contents.Count == 0 && u.ResponseId != null); // RunFinished has no text content - } - - [Fact] - public async Task RunStreamingAsync_WithNullMessages_ThrowsArgumentNullExceptionAsync() - { - // Arrange - using HttpClient httpClient = new(); - AGUIAgent agent = new("agent1", "Test agent", httpClient, "http://localhost/agent"); - - // Act & Assert - await Assert.ThrowsAsync(async () => - { - await foreach (var _ in agent.RunStreamingAsync(messages: null!)) - { - // Intentionally empty - consuming stream to trigger exception - } - }); - } - - [Fact] - public async Task RunStreamingAsync_WithNullThread_CreatesNewThreadAsync() - { - // Arrange - using HttpClient httpClient = this.CreateMockHttpClient(new BaseEvent[] - { - new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, - new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } - }); - - AGUIAgent agent = new("agent1", "Test agent", httpClient, "http://localhost/agent"); - List messages = [new ChatMessage(ChatRole.User, "Test")]; - - // Act - List updates = []; - await foreach (AgentRunResponseUpdate update in agent.RunStreamingAsync(messages, thread: null)) - { - // Consume the stream - updates.Add(update); - } - - // Assert - Assert.NotEmpty(updates); - } - - [Fact] - public async Task RunStreamingAsync_WithNonAGUIAgentThread_ThrowsInvalidOperationExceptionAsync() - { - // Arrange - using HttpClient httpClient = new(); - AGUIAgent agent = new("agent1", "Test agent", httpClient, "http://localhost/agent"); - List messages = [new ChatMessage(ChatRole.User, "Test")]; - AgentThread invalidThread = new TestInMemoryAgentThread(); - - // Act & Assert - await Assert.ThrowsAsync(async () => - { - await foreach (var _ in agent.RunStreamingAsync(messages, thread: invalidThread)) - { - // Consume the stream - } - }); - } - - [Fact] - public async Task RunStreamingAsync_GeneratesUniqueRunId_ForEachInvocationAsync() - { - // Arrange - List capturedRunIds = []; - using HttpClient httpClient = this.CreateMockHttpClientWithCapture(new BaseEvent[] - { - new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, - new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } - }, capturedRunIds); - - AGUIAgent agent = new("agent1", "Test agent", httpClient, "http://localhost/agent"); - List messages = [new ChatMessage(ChatRole.User, "Test")]; - - // Act - await foreach (var _ in agent.RunStreamingAsync(messages)) - { - // Consume the stream - } - await foreach (var _ in agent.RunStreamingAsync(messages)) - { - // Consume the stream - } - - // Assert - Assert.Equal(2, capturedRunIds.Count); - Assert.NotEqual(capturedRunIds[0], capturedRunIds[1]); - } - - [Fact] - public async Task RunStreamingAsync_NotifiesThreadOfNewMessages_AfterCompletionAsync() - { - // Arrange - using HttpClient httpClient = this.CreateMockHttpClient( - [ - new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, - new TextMessageStartEvent { MessageId = "msg1", Role = AGUIRoles.Assistant }, - new TextMessageContentEvent { MessageId = "msg1", Delta = "Hello" }, - new TextMessageEndEvent { MessageId = "msg1" }, - new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } - ]); - - AGUIAgent agent = new("agent1", "Test agent", httpClient, "http://localhost/agent"); - AGUIAgentThread thread = new(); - List messages = [new ChatMessage(ChatRole.User, "Test")]; - - // Act - await foreach (var _ in agent.RunStreamingAsync(messages, thread)) - { - // Consume the stream - } - - // Assert - Assert.NotEmpty(thread.MessageStore); - } - - [Fact] - public void DeserializeThread_WithValidState_ReturnsAGUIAgentThread() - { - // Arrange - using var httpClient = new HttpClient(); - AGUIAgent agent = new("agent1", "Test agent", httpClient, "http://localhost/agent"); - AGUIAgentThread originalThread = new() { ThreadId = "test-thread-123" }; - JsonElement serialized = originalThread.Serialize(); - - // Act - AgentThread deserialized = agent.DeserializeThread(serialized); - - // Assert - Assert.NotNull(deserialized); - Assert.IsType(deserialized); - AGUIAgentThread typedThread = (AGUIAgentThread)deserialized; - Assert.Equal("test-thread-123", typedThread.ThreadId); - } - - private HttpClient CreateMockHttpClient(BaseEvent[] events) - { - string sseContent = string.Join("", events.Select(e => - $"data: {JsonSerializer.Serialize(e, AGUIJsonSerializerContext.Default.BaseEvent)}\n\n")); - - Mock handlerMock = new(); - handlerMock - .Protected() - .Setup>( - "SendAsync", - ItExpr.IsAny(), - ItExpr.IsAny()) - .ReturnsAsync(new HttpResponseMessage - { - StatusCode = HttpStatusCode.OK, - Content = new StringContent(sseContent) - }); - - return new HttpClient(handlerMock.Object); - } - - private HttpClient CreateMockHttpClientWithCapture(BaseEvent[] events, List capturedRunIds) - { - string sseContent = string.Join("", events.Select(e => - $"data: {JsonSerializer.Serialize(e, AGUIJsonSerializerContext.Default.BaseEvent)}\n\n")); - - Mock handlerMock = new(); - handlerMock - .Protected() - .Setup>( - "SendAsync", - ItExpr.IsAny(), - ItExpr.IsAny()) - .Returns(async (HttpRequestMessage request, CancellationToken ct) => - { -#if NET - string requestBody = await request.Content!.ReadAsStringAsync(ct).ConfigureAwait(false); -#else - string requestBody = await request.Content!.ReadAsStringAsync().ConfigureAwait(false); -#endif - RunAgentInput? input = JsonSerializer.Deserialize(requestBody, AGUIJsonSerializerContext.Default.RunAgentInput); - if (input != null) - { - capturedRunIds.Add(input.RunId); - } - - return new HttpResponseMessage - { - StatusCode = HttpStatusCode.OK, - Content = new StringContent(sseContent) - }; - }); - - return new HttpClient(handlerMock.Object); - } - - private sealed class TestInMemoryAgentThread : InMemoryAgentThread - { - } -} diff --git a/dotnet/tests/Microsoft.Agents.AI.AGUI.UnitTests/AGUIAgentThreadTests.cs b/dotnet/tests/Microsoft.Agents.AI.AGUI.UnitTests/AGUIAgentThreadTests.cs deleted file mode 100644 index 1ddc39cdfc..0000000000 --- a/dotnet/tests/Microsoft.Agents.AI.AGUI.UnitTests/AGUIAgentThreadTests.cs +++ /dev/null @@ -1,132 +0,0 @@ -// Copyright (c) Microsoft. All rights reserved. - -using System; -using System.Linq; -using System.Text.Json; -using System.Threading; -using System.Threading.Tasks; -using Microsoft.Extensions.AI; - -namespace Microsoft.Agents.AI.AGUI.UnitTests; - -public sealed class AGUIAgentThreadTests -{ - [Fact] - public void Constructor_WithValidThreadId_DeserializesSuccessfully() - { - // Arrange - const string ThreadId = "thread123"; - AGUIAgentThread originalThread = new() { ThreadId = ThreadId }; - JsonElement serialized = originalThread.Serialize(); - - // Act - AGUIAgentThread deserializedThread = new(serialized); - - // Assert - Assert.Equal(ThreadId, deserializedThread.ThreadId); - } - - [Fact] - public void Constructor_WithMissingThreadId_ThrowsInvalidOperationException() - { - // Arrange - const string Json = """ - {"WrappedState":{}} - """; - JsonElement serialized = JsonSerializer.Deserialize(Json); - - // Act & Assert - Assert.Throws(() => new AGUIAgentThread(serialized)); - } - - [Fact] - public void Constructor_WithMissingWrappedState_ThrowsArgumentException() - { - // Arrange - const string Json = """ - {} - """; - JsonElement serialized = JsonSerializer.Deserialize(Json); - - // Act & Assert - Assert.Throws(() => new AGUIAgentThread(serialized)); - } - - [Fact] - public async Task Constructor_UnwrapsAndRestores_BaseStateAsync() - { - // Arrange - AGUIAgentThread originalThread = new() { ThreadId = "thread1" }; - ChatMessage message = new(ChatRole.User, "Test message"); - await TestAgent.AddMessageToThreadAsync(originalThread, message); - JsonElement serialized = originalThread.Serialize(); - - // Act - AGUIAgentThread deserializedThread = new(serialized); - - // Assert - Assert.Single(deserializedThread.MessageStore); - Assert.Equal("Test message", deserializedThread.MessageStore.First().Text); - } - - [Fact] - public void Serialize_IncludesThreadId_InSerializedState() - { - // Arrange - const string ThreadId = "thread456"; - AGUIAgentThread thread = new() { ThreadId = ThreadId }; - - // Act - JsonElement serialized = thread.Serialize(); - - // Assert - Assert.True(serialized.TryGetProperty("ThreadId", out JsonElement threadIdElement)); - Assert.Equal(ThreadId, threadIdElement.GetString()); - } - - [Fact] - public async Task Serialize_WrapsBaseState_CorrectlyAsync() - { - // Arrange - AGUIAgentThread thread = new() { ThreadId = "thread1" }; - ChatMessage message = new(ChatRole.User, "Test message"); - await TestAgent.AddMessageToThreadAsync(thread, message); - - // Act - JsonElement serialized = thread.Serialize(); - - // Assert - Assert.True(serialized.TryGetProperty("WrappedState", out JsonElement wrappedState)); - Assert.NotEqual(JsonValueKind.Null, wrappedState.ValueKind); - } - - [Fact] - public async Task Serialize_RoundTrip_PreservesThreadIdAndMessagesAsync() - { - // Arrange - const string ThreadId = "thread789"; - AGUIAgentThread originalThread = new() { ThreadId = ThreadId }; - ChatMessage message1 = new(ChatRole.User, "First message"); - ChatMessage message2 = new(ChatRole.Assistant, "Second message"); - await TestAgent.AddMessageToThreadAsync(originalThread, message1); - await TestAgent.AddMessageToThreadAsync(originalThread, message2); - - // Act - JsonElement serialized = originalThread.Serialize(); - AGUIAgentThread deserializedThread = new(serialized); - - // Assert - Assert.Equal(ThreadId, deserializedThread.ThreadId); - Assert.Equal(2, deserializedThread.MessageStore.Count); - Assert.Equal("First message", deserializedThread.MessageStore.ElementAt(0).Text); - Assert.Equal("Second message", deserializedThread.MessageStore.ElementAt(1).Text); - } - - private abstract class TestAgent : AIAgent - { - public static async Task AddMessageToThreadAsync(AgentThread thread, ChatMessage message) - { - await NotifyThreadOfNewMessagesAsync(thread, [message], CancellationToken.None); - } - } -} diff --git a/dotnet/tests/Microsoft.Agents.AI.AGUI.UnitTests/AGUIChatClientTests.cs b/dotnet/tests/Microsoft.Agents.AI.AGUI.UnitTests/AGUIChatClientTests.cs new file mode 100644 index 0000000000..c61a7e289d --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.AGUI.UnitTests/AGUIChatClientTests.cs @@ -0,0 +1,1739 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Collections.Generic; +using System.Linq; +using System.Net; +using System.Net.Http; +using System.Text.Json; +using System.Threading; +using System.Threading.Tasks; +using Microsoft.Agents.AI.AGUI.Shared; +using Microsoft.Extensions.AI; + +namespace Microsoft.Agents.AI.AGUI.UnitTests; + +public sealed class AGUIAgentTests +{ + [Fact] + public async Task RunAsync_AggregatesStreamingUpdates_ReturnsCompleteMessagesAsync() + { + // Arrange + using HttpClient httpClient = this.CreateMockHttpClient( + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, + new TextMessageStartEvent { MessageId = "msg1", Role = AGUIRoles.Assistant }, + new TextMessageContentEvent { MessageId = "msg1", Delta = "Hello" }, + new TextMessageContentEvent { MessageId = "msg1", Delta = " World" }, + new TextMessageEndEvent { MessageId = "msg1" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } + ]); + + var chatClient = new AGUIChatClient(httpClient, "http://localhost/agent", null, AGUIJsonSerializerContext.Default.Options); + AIAgent agent = chatClient.CreateAIAgent(instructions: null, name: "agent1", description: "Test agent", tools: []); + List messages = [new ChatMessage(ChatRole.User, "Test")]; + + // Act + AgentRunResponse response = await agent.RunAsync(messages); + + // Assert + Assert.NotNull(response); + Assert.NotEmpty(response.Messages); + ChatMessage message = response.Messages.First(); + Assert.Equal(ChatRole.Assistant, message.Role); + Assert.Equal("Hello World", message.Text); + } + + [Fact] + public async Task RunAsync_WithEmptyUpdateStream_ContainsOnlyMetadataMessagesAsync() + { + // Arrange + using HttpClient httpClient = this.CreateMockHttpClient( + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } + ]); + + var chatClient = new AGUIChatClient(httpClient, "http://localhost/agent", null, AGUIJsonSerializerContext.Default.Options); + AIAgent agent = chatClient.CreateAIAgent(instructions: null, name: "agent1", description: "Test agent", tools: []); + List messages = [new ChatMessage(ChatRole.User, "Test")]; + + // Act + AgentRunResponse response = await agent.RunAsync(messages); + + // Assert + Assert.NotNull(response); + // RunStarted and RunFinished events are aggregated into messages by ToChatResponse() + Assert.NotEmpty(response.Messages); + Assert.All(response.Messages, m => Assert.Equal(ChatRole.Assistant, m.Role)); + } + + [Fact] + public async Task RunAsync_WithNullMessages_ThrowsArgumentNullExceptionAsync() + { + // Arrange + using HttpClient httpClient = new(); + var chatClient = new AGUIChatClient(httpClient, "http://localhost/agent", null, AGUIJsonSerializerContext.Default.Options); + AIAgent agent = chatClient.CreateAIAgent(instructions: "Test agent", name: "agent1"); + + // Act & Assert + await Assert.ThrowsAsync(() => agent.RunAsync(messages: null!)); + } + + [Fact] + public async Task RunAsync_WithNullThread_CreatesNewThreadAsync() + { + // Arrange + using HttpClient httpClient = this.CreateMockHttpClient( + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } + ]); + + var chatClient = new AGUIChatClient(httpClient, "http://localhost/agent", null, AGUIJsonSerializerContext.Default.Options); + AIAgent agent = chatClient.CreateAIAgent(instructions: "Test agent", name: "agent1"); + List messages = [new ChatMessage(ChatRole.User, "Test")]; + + // Act + AgentRunResponse response = await agent.RunAsync(messages, thread: null); + + // Assert + Assert.NotNull(response); + } + + [Fact] + public async Task RunStreamingAsync_YieldsAllEvents_FromServerStreamAsync() + { + // Arrange + using HttpClient httpClient = this.CreateMockHttpClient( + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, + new TextMessageStartEvent { MessageId = "msg1", Role = AGUIRoles.Assistant }, + new TextMessageContentEvent { MessageId = "msg1", Delta = "Hello" }, + new TextMessageEndEvent { MessageId = "msg1" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } + ]); + + var chatClient = new AGUIChatClient(httpClient, "http://localhost/agent", null, AGUIJsonSerializerContext.Default.Options); + AIAgent agent = chatClient.CreateAIAgent(instructions: "Test agent", name: "agent1"); + List messages = [new ChatMessage(ChatRole.User, "Test")]; + + // Act + List updates = []; + await foreach (AgentRunResponseUpdate update in agent.RunStreamingAsync(messages)) + { + // Consume the stream + updates.Add(update); + } + + // Assert + Assert.NotEmpty(updates); + Assert.Contains(updates, u => u.ResponseId != null); // RunStarted sets ResponseId + Assert.Contains(updates, u => u.Contents.Any(c => c is TextContent)); + Assert.Contains(updates, u => u.Contents.Count == 0 && u.ResponseId != null); // RunFinished has no text content + } + + [Fact] + public async Task RunStreamingAsync_WithNullMessages_ThrowsArgumentNullExceptionAsync() + { + // Arrange + using HttpClient httpClient = new(); + var chatClient = new AGUIChatClient(httpClient, "http://localhost/agent", null, AGUIJsonSerializerContext.Default.Options); + AIAgent agent = chatClient.CreateAIAgent(instructions: "Test agent", name: "agent1"); + + // Act & Assert + await Assert.ThrowsAsync(async () => + { + await foreach (var _ in agent.RunStreamingAsync(messages: null!)) + { + // Intentionally empty - consuming stream to trigger exception + } + }); + } + + [Fact] + public async Task RunStreamingAsync_WithNullThread_CreatesNewThreadAsync() + { + // Arrange + using HttpClient httpClient = this.CreateMockHttpClient( + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } + ]); + + var chatClient = new AGUIChatClient(httpClient, "http://localhost/agent", null, AGUIJsonSerializerContext.Default.Options); + AIAgent agent = chatClient.CreateAIAgent(instructions: "Test agent", name: "agent1"); + List messages = [new ChatMessage(ChatRole.User, "Test")]; + + // Act + List updates = []; + await foreach (AgentRunResponseUpdate update in agent.RunStreamingAsync(messages, thread: null)) + { + // Consume the stream + updates.Add(update); + } + + // Assert + Assert.NotEmpty(updates); + } + + [Fact] + public async Task RunStreamingAsync_GeneratesUniqueRunId_ForEachInvocationAsync() + { + // Arrange + var handler = new TestDelegatingHandler(); + handler.AddResponseWithCapture( + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } + ]); + handler.AddResponseWithCapture( + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run2" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run2" } + ]); + using HttpClient httpClient = new(handler); + + var chatClient = new AGUIChatClient(httpClient, "http://localhost/agent", null, AGUIJsonSerializerContext.Default.Options); + AIAgent agent = chatClient.CreateAIAgent(instructions: null, name: "agent1", description: "Test agent", tools: []); + List messages = [new ChatMessage(ChatRole.User, "Test")]; + + // Act + await foreach (var _ in agent.RunStreamingAsync(messages)) + { + // Consume the stream + } + await foreach (var _ in agent.RunStreamingAsync(messages)) + { + // Consume the stream + } + + // Assert + Assert.Equal(2, handler.CapturedRunIds.Count); + Assert.NotEqual(handler.CapturedRunIds[0], handler.CapturedRunIds[1]); + } + + [Fact] + public async Task RunStreamingAsync_ReturnsStreamingUpdates_AfterCompletionAsync() + { + // Arrange + using HttpClient httpClient = this.CreateMockHttpClient( + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, + new TextMessageStartEvent { MessageId = "msg1", Role = AGUIRoles.Assistant }, + new TextMessageContentEvent { MessageId = "msg1", Delta = "Hello" }, + new TextMessageEndEvent { MessageId = "msg1" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } + ]); + + var chatClient = new AGUIChatClient(httpClient, "http://localhost/agent", null, AGUIJsonSerializerContext.Default.Options); + AIAgent agent = chatClient.CreateAIAgent(instructions: null, name: "agent1", description: "Test agent", tools: []); + AgentThread thread = agent.GetNewThread(); + List messages = [new ChatMessage(ChatRole.User, "Hello")]; + + // Act + List updates = []; + await foreach (var update in agent.RunStreamingAsync(messages, thread)) + { + updates.Add(update); + } + + // Assert - Verify streaming updates were received + Assert.NotEmpty(updates); + Assert.Contains(updates, u => u.Text == "Hello"); + } + + [Fact] + public void DeserializeThread_WithValidState_ReturnsChatClientAgentThread() + { + // Arrange + using var httpClient = new HttpClient(); + var chatClient = new AGUIChatClient(httpClient, "http://localhost/agent", null, AGUIJsonSerializerContext.Default.Options); + AIAgent agent = chatClient.CreateAIAgent(instructions: null, name: "agent1", description: "Test agent", tools: []); + AgentThread originalThread = agent.GetNewThread(); + JsonElement serialized = originalThread.Serialize(); + + // Act + AgentThread deserialized = agent.DeserializeThread(serialized); + + // Assert + Assert.NotNull(deserialized); + Assert.IsType(deserialized); + } + + private HttpClient CreateMockHttpClient(BaseEvent[] events) + { + var handler = new TestDelegatingHandler(); + handler.AddResponse(events); + return new HttpClient(handler); + } + + [Fact] + public async Task RunStreamingAsync_InvokesTools_WhenFunctionCallsReturnedAsync() + { + // Arrange + bool toolInvoked = false; + AIFunction testTool = AIFunctionFactory.Create( + (string location) => + { + toolInvoked = true; + return $"Weather in {location}: Sunny, 72°F"; + }, + "GetWeather", + "Gets the current weather for a location"); + + using HttpClient httpClient = this.CreateMockHttpClientForToolCalls( + firstResponse: + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, + new ToolCallStartEvent { ToolCallId = "call_1", ToolCallName = "GetWeather", ParentMessageId = "msg1" }, + new ToolCallArgsEvent { ToolCallId = "call_1", Delta = "{\"location\":\"Seattle\"}" }, + new ToolCallEndEvent { ToolCallId = "call_1" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } + ], + secondResponse: + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run2" }, + new TextMessageStartEvent { MessageId = "msg2", Role = AGUIRoles.Assistant }, + new TextMessageContentEvent { MessageId = "msg2", Delta = "The weather is nice!" }, + new TextMessageEndEvent { MessageId = "msg2" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run2" } + ]); + + var chatClient = new AGUIChatClient(httpClient, "http://localhost/agent", null, AGUIJsonSerializerContext.Default.Options); + AIAgent agent = chatClient.CreateAIAgent(instructions: null, name: "agent1", description: "Test agent", tools: [testTool]); + List messages = [new ChatMessage(ChatRole.User, "What's the weather?")]; + + // Act + List allUpdates = []; + await foreach (AgentRunResponseUpdate update in agent.RunStreamingAsync(messages)) + { + allUpdates.Add(update); + } + + // Assert + Assert.True(toolInvoked, "Tool should have been invoked"); + Assert.NotEmpty(allUpdates); + // Should have updates from both the tool call and the final response + Assert.Contains(allUpdates, u => u.Contents.Any(c => c is FunctionCallContent)); + Assert.Contains(allUpdates, u => u.Contents.Any(c => c is TextContent)); + } + + [Fact] + public async Task RunStreamingAsync_DoesNotInvokeTools_WhenSomeToolsNotAvailableAsync() + { + // Arrange + bool tool1Invoked = false; + AIFunction tool1 = AIFunctionFactory.Create( + () => { tool1Invoked = true; return "Result1"; }, + "Tool1"); + + // FunctionInvokingChatClient makes two calls: first gets tool calls, second returns final response + // When not all tools are available, it invokes the ones that ARE available + var handler = new TestDelegatingHandler(); + handler.AddResponse( + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, + new ToolCallStartEvent { ToolCallId = "call_1", ToolCallName = "Tool1", ParentMessageId = "msg1" }, + new ToolCallArgsEvent { ToolCallId = "call_1", Delta = "{}" }, + new ToolCallEndEvent { ToolCallId = "call_1" }, + new ToolCallStartEvent { ToolCallId = "call_2", ToolCallName = "Tool2", ParentMessageId = "msg1" }, + new ToolCallArgsEvent { ToolCallId = "call_2", Delta = "{}" }, + new ToolCallEndEvent { ToolCallId = "call_2" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } + ]); + handler.AddResponse( + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run2" }, + new TextMessageStartEvent { MessageId = "msg2", Role = AGUIRoles.Assistant }, + new TextMessageContentEvent { MessageId = "msg2", Delta = "Response" }, + new TextMessageEndEvent { MessageId = "msg2" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run2" } + ]); + using HttpClient httpClient = new(handler); + + var chatClient = new AGUIChatClient(httpClient, "http://localhost/agent", null, AGUIJsonSerializerContext.Default.Options); + AIAgent agent = chatClient.CreateAIAgent(instructions: null, name: "agent1", description: "Test agent", tools: [tool1]); // Only tool1, not tool2 + List messages = [new ChatMessage(ChatRole.User, "Test")]; + + // Act + List allUpdates = []; + await foreach (AgentRunResponseUpdate update in agent.RunStreamingAsync(messages)) + { + allUpdates.Add(update); + } + + // Assert + // FunctionInvokingChatClient invokes Tool1 since it's available, even though Tool2 is not + Assert.True(tool1Invoked, "Tool1 should be invoked even though Tool2 is not available"); + // Should have tool call results for Tool1 and an error result for Tool2 + Assert.Contains(allUpdates, u => u.Contents.Any(c => c is FunctionResultContent frc && frc.CallId == "call_1")); + } + + [Fact] + public async Task RunStreamingAsync_HandlesToolInvocationErrors_GracefullyAsync() + { + // Arrange + AIFunction faultyTool = AIFunctionFactory.Create( + () => + { + throw new InvalidOperationException("Tool failed!"); +#pragma warning disable CS0162 // Unreachable code detected + return string.Empty; +#pragma warning restore CS0162 // Unreachable code detected + }, + "FaultyTool"); + + using HttpClient httpClient = this.CreateMockHttpClientForToolCalls( + firstResponse: + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, + new ToolCallStartEvent { ToolCallId = "call_1", ToolCallName = "FaultyTool", ParentMessageId = "msg1" }, + new ToolCallArgsEvent { ToolCallId = "call_1", Delta = "{}" }, + new ToolCallEndEvent { ToolCallId = "call_1" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } + ], + secondResponse: + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run2" }, + new TextMessageStartEvent { MessageId = "msg2", Role = AGUIRoles.Assistant }, + new TextMessageContentEvent { MessageId = "msg2", Delta = "I encountered an error." }, + new TextMessageEndEvent { MessageId = "msg2" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run2" } + ]); + + var chatClient = new AGUIChatClient(httpClient, "http://localhost/agent", null, AGUIJsonSerializerContext.Default.Options); + AIAgent agent = chatClient.CreateAIAgent(instructions: null, name: "agent1", description: "Test agent", tools: [faultyTool]); + List messages = [new ChatMessage(ChatRole.User, "Test")]; + + // Act + List allUpdates = []; + await foreach (AgentRunResponseUpdate update in agent.RunStreamingAsync(messages)) + { + allUpdates.Add(update); + } + + // Assert - should complete without throwing + Assert.NotEmpty(allUpdates); + } + + [Fact] + public async Task RunStreamingAsync_InvokesMultipleTools_InSingleTurnAsync() + { + // Arrange + int tool1CallCount = 0; + int tool2CallCount = 0; + AIFunction tool1 = AIFunctionFactory.Create(() => { tool1CallCount++; return "Result1"; }, "Tool1"); + AIFunction tool2 = AIFunctionFactory.Create(() => { tool2CallCount++; return "Result2"; }, "Tool2"); + + using HttpClient httpClient = this.CreateMockHttpClientForToolCalls( + firstResponse: + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, + new ToolCallStartEvent { ToolCallId = "call_1", ToolCallName = "Tool1", ParentMessageId = "msg1" }, + new ToolCallArgsEvent { ToolCallId = "call_1", Delta = "{}" }, + new ToolCallEndEvent { ToolCallId = "call_1" }, + new ToolCallStartEvent { ToolCallId = "call_2", ToolCallName = "Tool2", ParentMessageId = "msg1" }, + new ToolCallArgsEvent { ToolCallId = "call_2", Delta = "{}" }, + new ToolCallEndEvent { ToolCallId = "call_2" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } + ], + secondResponse: + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run2" }, + new TextMessageStartEvent { MessageId = "msg2", Role = AGUIRoles.Assistant }, + new TextMessageContentEvent { MessageId = "msg2", Delta = "Done" }, + new TextMessageEndEvent { MessageId = "msg2" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run2" } + ]); + + var chatClient = new AGUIChatClient(httpClient, "http://localhost/agent", null, AGUIJsonSerializerContext.Default.Options); + AIAgent agent = chatClient.CreateAIAgent(instructions: null, name: "agent1", description: "Test agent", tools: [tool1, tool2]); + List messages = [new ChatMessage(ChatRole.User, "Test")]; + + // Act + await foreach (var _ in agent.RunStreamingAsync(messages)) + { + } + + // Assert + Assert.Equal(1, tool1CallCount); + Assert.Equal(1, tool2CallCount); + } + + [Fact] + public async Task RunStreamingAsync_UpdatesThreadWithToolMessages_AfterCompletionAsync() + { + // Arrange + AIFunction testTool = AIFunctionFactory.Create(() => "Result", "TestTool"); + + using HttpClient httpClient = this.CreateMockHttpClientForToolCalls( + firstResponse: + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, + new ToolCallStartEvent { ToolCallId = "call_1", ToolCallName = "TestTool", ParentMessageId = "msg1" }, + new ToolCallArgsEvent { ToolCallId = "call_1", Delta = "{}" }, + new ToolCallEndEvent { ToolCallId = "call_1" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } + ], + secondResponse: + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run2" }, + new TextMessageStartEvent { MessageId = "msg2", Role = AGUIRoles.Assistant }, + new TextMessageContentEvent { MessageId = "msg2", Delta = "Complete" }, + new TextMessageEndEvent { MessageId = "msg2" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run2" } + ]); + + var chatClient = new AGUIChatClient(httpClient, "http://localhost/agent", null, AGUIJsonSerializerContext.Default.Options); + AIAgent agent = chatClient.CreateAIAgent(instructions: null, name: "agent1", description: "Test agent", tools: [testTool]); + AgentThread thread = agent.GetNewThread(); + List messages = [new ChatMessage(ChatRole.User, "Test")]; + + // Act + List updates = []; + await foreach (var update in agent.RunStreamingAsync(messages, thread)) + { + updates.Add(update); + } + + // Assert - Verify we received updates including tool calls + Assert.NotEmpty(updates); + Assert.Contains(updates, u => u.Contents.Any(c => c is FunctionCallContent)); + Assert.Contains(updates, u => u.Contents.Any(c => c is FunctionResultContent)); + Assert.Contains(updates, u => u.Text == "Complete"); + } + + private HttpClient CreateMockHttpClientForToolCalls(BaseEvent[] firstResponse, BaseEvent[] secondResponse) + { + var handler = new TestDelegatingHandler(); + handler.AddResponse(firstResponse); + handler.AddResponse(secondResponse); + return new HttpClient(handler); + } + + [Fact] + public async Task GetStreamingResponseAsync_WrapsServerFunctionCalls_InServerFunctionCallContentAsync() + { + // Arrange - Server returns a function call for a tool not in the client tool set + using HttpClient httpClient = this.CreateMockHttpClient( + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, + new ToolCallStartEvent { ToolCallId = "call_1", ToolCallName = "ServerTool", ParentMessageId = "msg1" }, + new ToolCallArgsEvent { ToolCallId = "call_1", Delta = "{\"arg\":\"value\"}" }, + new ToolCallEndEvent { ToolCallId = "call_1" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } + ]); + + var chatClient = new AGUIChatClient(httpClient, "http://localhost/agent", null, AGUIJsonSerializerContext.Default.Options); + // No tools provided - any function call from server is a "server function" + var options = new ChatOptions(); + List messages = [new ChatMessage(ChatRole.User, "Test")]; + + // Act + List updates = []; + await foreach (var update in chatClient.GetStreamingResponseAsync(messages, options)) + { + updates.Add(update); + } + + // Assert - Server function call should be presented as FunctionCallContent (unwrapped) + Assert.Contains(updates, u => u.Contents.Any(c => c is FunctionCallContent fcc && fcc.Name == "ServerTool")); + // Should NOT contain ServerFunctionCallContent (it's internal and unwrapped before yielding) + Assert.DoesNotContain(updates, u => u.Contents.Any(c => c.GetType().Name == "ServerFunctionCallContent")); + } + + [Fact] + public async Task GetStreamingResponseAsync_DoesNotWrapClientFunctionCalls_WhenToolInClientSetAsync() + { + // Arrange + AIFunction clientTool = AIFunctionFactory.Create(() => "Result", "ClientTool"); + + var handler = new TestDelegatingHandler(); + handler.AddResponse( + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, + new ToolCallStartEvent { ToolCallId = "call_1", ToolCallName = "ClientTool", ParentMessageId = "msg1" }, + new ToolCallArgsEvent { ToolCallId = "call_1", Delta = "{}" }, + new ToolCallEndEvent { ToolCallId = "call_1" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } + ]); + handler.AddResponse( + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run2" }, + new TextMessageStartEvent { MessageId = "msg2", Role = AGUIRoles.Assistant }, + new TextMessageContentEvent { MessageId = "msg2", Delta = "Done" }, + new TextMessageEndEvent { MessageId = "msg2" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run2" } + ]); + using HttpClient httpClient = new(handler); + + var chatClient = new AGUIChatClient(httpClient, "http://localhost/agent", null, AGUIJsonSerializerContext.Default.Options); + var options = new ChatOptions { Tools = [clientTool] }; + List messages = [new ChatMessage(ChatRole.User, "Test")]; + + // Act + List updates = []; + await foreach (var update in chatClient.GetStreamingResponseAsync(messages, options)) + { + updates.Add(update); + } + + // Assert - Should have function call and result (FunctionInvokingChatClient processed it) + Assert.Contains(updates, u => u.Contents.Any(c => c is FunctionCallContent fcc && fcc.Name == "ClientTool")); + Assert.Contains(updates, u => u.Contents.Any(c => c is FunctionResultContent frc && frc.CallId == "call_1")); + } + + [Fact] + public async Task GetStreamingResponseAsync_HandlesMixedClientAndServerFunctions_InSameResponseAsync() + { + // Arrange + AIFunction clientTool = AIFunctionFactory.Create(() => "ClientResult", "ClientTool"); + + var handler = new TestDelegatingHandler(); + handler.AddResponse( + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, + new ToolCallStartEvent { ToolCallId = "call_1", ToolCallName = "ClientTool", ParentMessageId = "msg1" }, + new ToolCallArgsEvent { ToolCallId = "call_1", Delta = "{}" }, + new ToolCallEndEvent { ToolCallId = "call_1" }, + new ToolCallStartEvent { ToolCallId = "call_2", ToolCallName = "ServerTool", ParentMessageId = "msg1" }, + new ToolCallArgsEvent { ToolCallId = "call_2", Delta = "{}" }, + new ToolCallEndEvent { ToolCallId = "call_2" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } + ]); + handler.AddResponse( + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run2" }, + new TextMessageStartEvent { MessageId = "msg2", Role = AGUIRoles.Assistant }, + new TextMessageContentEvent { MessageId = "msg2", Delta = "Done" }, + new TextMessageEndEvent { MessageId = "msg2" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run2" } + ]); + using HttpClient httpClient = new(handler); + + var chatClient = new AGUIChatClient(httpClient, "http://localhost/agent", null, AGUIJsonSerializerContext.Default.Options); + var options = new ChatOptions { Tools = [clientTool] }; + List messages = [new ChatMessage(ChatRole.User, "Test")]; + + // Act + List updates = []; + await foreach (var update in chatClient.GetStreamingResponseAsync(messages, options)) + { + updates.Add(update); + } + + // Assert - Should have both client and server function calls + Assert.Contains(updates, u => u.Contents.Any(c => c is FunctionCallContent fcc && fcc.Name == "ClientTool")); + Assert.Contains(updates, u => u.Contents.Any(c => c is FunctionCallContent fcc && fcc.Name == "ServerTool")); + // Client tool should have result + Assert.Contains(updates, u => u.Contents.Any(c => c is FunctionResultContent frc && frc.CallId == "call_1")); + } + + [Fact] + public async Task GetStreamingResponseAsync_PreservesConversationId_AcrossMultipleTurnsAsync() + { + // Arrange + var handler = new TestDelegatingHandler(); + handler.AddResponse( + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, + new TextMessageStartEvent { MessageId = "msg1", Role = AGUIRoles.Assistant }, + new TextMessageContentEvent { MessageId = "msg1", Delta = "First" }, + new TextMessageEndEvent { MessageId = "msg1" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } + ]); + handler.AddResponse( + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run2" }, + new TextMessageStartEvent { MessageId = "msg2", Role = AGUIRoles.Assistant }, + new TextMessageContentEvent { MessageId = "msg2", Delta = "Second" }, + new TextMessageEndEvent { MessageId = "msg2" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run2" } + ]); + using HttpClient httpClient = new(handler); + + var chatClient = new AGUIChatClient(httpClient, "http://localhost/agent", null, AGUIJsonSerializerContext.Default.Options); + var options = new ChatOptions { ConversationId = "my-conversation-123" }; + List messages = [new ChatMessage(ChatRole.User, "Test")]; + + // Act - First turn + List updates1 = []; + await foreach (var update in chatClient.GetStreamingResponseAsync(messages, options)) + { + updates1.Add(update); + } + + // Second turn with same conversation ID + List updates2 = []; + await foreach (var update in chatClient.GetStreamingResponseAsync(messages, options)) + { + updates2.Add(update); + } + + // Assert - Both turns should preserve the conversation ID + Assert.All(updates1, u => Assert.Equal("my-conversation-123", u.ConversationId)); + Assert.All(updates2, u => Assert.Equal("my-conversation-123", u.ConversationId)); + } + + [Fact] + public async Task GetStreamingResponseAsync_ExtractsThreadId_FromServerResponseAsync() + { + // Arrange + using HttpClient httpClient = this.CreateMockHttpClient( + [ + new RunStartedEvent { ThreadId = "server-thread-456", RunId = "run1" }, + new TextMessageStartEvent { MessageId = "msg1", Role = AGUIRoles.Assistant }, + new TextMessageContentEvent { MessageId = "msg1", Delta = "Hello" }, + new TextMessageEndEvent { MessageId = "msg1" }, + new RunFinishedEvent { ThreadId = "server-thread-456", RunId = "run1" } + ]); + + var chatClient = new AGUIChatClient(httpClient, "http://localhost/agent", null, AGUIJsonSerializerContext.Default.Options); + // No conversation ID provided + List messages = [new ChatMessage(ChatRole.User, "Test")]; + + // Act + List updates = []; + await foreach (var update in chatClient.GetStreamingResponseAsync(messages, null)) + { + updates.Add(update); + } + + // Assert - Should use thread ID from server + Assert.All(updates, u => Assert.Equal("server-thread-456", u.ConversationId)); + } + + [Fact] + public async Task GetStreamingResponseAsync_GeneratesThreadId_WhenNoneProvidedAsync() + { + // Arrange + using HttpClient httpClient = this.CreateMockHttpClient( + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, + new TextMessageStartEvent { MessageId = "msg1", Role = AGUIRoles.Assistant }, + new TextMessageContentEvent { MessageId = "msg1", Delta = "Hello" }, + new TextMessageEndEvent { MessageId = "msg1" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } + ]); + + var chatClient = new AGUIChatClient(httpClient, "http://localhost/agent", null, AGUIJsonSerializerContext.Default.Options); + List messages = [new ChatMessage(ChatRole.User, "Test")]; + + // Act + List updates = []; + await foreach (var update in chatClient.GetStreamingResponseAsync(messages, null)) + { + updates.Add(update); + } + + // Assert - Should have a conversation ID (either from server or generated) + Assert.All(updates, u => Assert.NotNull(u.ConversationId)); + Assert.All(updates, u => Assert.NotEmpty(u.ConversationId!)); + } + + [Fact] + public async Task GetStreamingResponseAsync_RemovesThreadIdFromFunctionCallProperties_BeforeYieldingAsync() + { + // Arrange + AIFunction clientTool = AIFunctionFactory.Create(() => "Result", "ClientTool"); + + var handler = new TestDelegatingHandler(); + handler.AddResponse( + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, + new ToolCallStartEvent { ToolCallId = "call_1", ToolCallName = "ClientTool", ParentMessageId = "msg1" }, + new ToolCallArgsEvent { ToolCallId = "call_1", Delta = "{}" }, + new ToolCallEndEvent { ToolCallId = "call_1" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } + ]); + handler.AddResponse( + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run2" }, + new TextMessageStartEvent { MessageId = "msg2", Role = AGUIRoles.Assistant }, + new TextMessageContentEvent { MessageId = "msg2", Delta = "Done" }, + new TextMessageEndEvent { MessageId = "msg2" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run2" } + ]); + using HttpClient httpClient = new(handler); + + var chatClient = new AGUIChatClient(httpClient, "http://localhost/agent", null, AGUIJsonSerializerContext.Default.Options); + var options = new ChatOptions { Tools = [clientTool] }; + List messages = [new ChatMessage(ChatRole.User, "Test")]; + + // Act + List updates = []; + await foreach (var update in chatClient.GetStreamingResponseAsync(messages, options)) + { + updates.Add(update); + } + + // Assert - Function call content should not have agui_thread_id in additional properties + var functionCallUpdate = updates.FirstOrDefault(u => u.Contents.Any(c => c is FunctionCallContent)); + Assert.NotNull(functionCallUpdate); + var fcc = functionCallUpdate.Contents.OfType().First(); + Assert.True(fcc.AdditionalProperties?.ContainsKey("agui_thread_id") != true); + } + + [Fact] + public async Task GetResponseAsync_PreservesConversationId_ThroughStreamingPathAsync() + { + // Arrange + using HttpClient httpClient = this.CreateMockHttpClient( + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, + new TextMessageStartEvent { MessageId = "msg1", Role = AGUIRoles.Assistant }, + new TextMessageContentEvent { MessageId = "msg1", Delta = "Hello" }, + new TextMessageEndEvent { MessageId = "msg1" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } + ]); + + var chatClient = new AGUIChatClient(httpClient, "http://localhost/agent", null, AGUIJsonSerializerContext.Default.Options); + var options = new ChatOptions { ConversationId = "my-conversation-456" }; + List messages = [new ChatMessage(ChatRole.User, "Test")]; + + // Act + ChatResponse response = await chatClient.GetResponseAsync(messages, options); + + // Assert + Assert.Equal("my-conversation-456", response.ConversationId); + } + + [Fact] + public async Task GetStreamingResponseAsync_UsesServerThreadId_WhenDifferentFromClientAsync() + { + // Arrange - Server returns different thread ID + using HttpClient httpClient = this.CreateMockHttpClient( + [ + new RunStartedEvent { ThreadId = "server-generated-thread", RunId = "run1" }, + new TextMessageStartEvent { MessageId = "msg1", Role = AGUIRoles.Assistant }, + new TextMessageContentEvent { MessageId = "msg1", Delta = "Hello" }, + new TextMessageEndEvent { MessageId = "msg1" }, + new RunFinishedEvent { ThreadId = "server-generated-thread", RunId = "run1" } + ]); + + var chatClient = new AGUIChatClient(httpClient, "http://localhost/agent", null, AGUIJsonSerializerContext.Default.Options); + var options = new ChatOptions { ConversationId = "client-thread-123" }; + List messages = [new ChatMessage(ChatRole.User, "Test")]; + + // Act + List updates = []; + await foreach (var update in chatClient.GetStreamingResponseAsync(messages, options)) + { + updates.Add(update); + } + + // Assert - Should use client's conversation ID (we provided it explicitly) + Assert.All(updates, u => Assert.Equal("client-thread-123", u.ConversationId)); + } + + [Fact] + public async Task GetStreamingResponseAsync_FullConversationFlow_WithMixedFunctionsAsync() + { + // Arrange + AIFunction clientTool = AIFunctionFactory.Create(() => "ClientResult", "ClientTool"); + + var handler = new TestDelegatingHandler(); + // First response: client function call (FunctionInvokingChatClient will handle this) + handler.AddResponse( + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, + new ToolCallStartEvent { ToolCallId = "call_client", ToolCallName = "ClientTool", ParentMessageId = "msg1" }, + new ToolCallArgsEvent { ToolCallId = "call_client", Delta = "{}" }, + new ToolCallEndEvent { ToolCallId = "call_client" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } + ]); + // Second response: after client function execution, return final text + handler.AddResponse( + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run2" }, + new TextMessageStartEvent { MessageId = "msg2", Role = AGUIRoles.Assistant }, + new TextMessageContentEvent { MessageId = "msg2", Delta = "Complete" }, + new TextMessageEndEvent { MessageId = "msg2" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run2" } + ]); + using HttpClient httpClient = new(handler); + + var chatClient = new AGUIChatClient(httpClient, "http://localhost/agent", null, AGUIJsonSerializerContext.Default.Options); + var options = new ChatOptions { Tools = [clientTool] }; + List messages = [new ChatMessage(ChatRole.User, "Test")]; + + // Act + List updates = []; + string? conversationId = null; + await foreach (var update in chatClient.GetStreamingResponseAsync(messages, options)) + { + updates.Add(update); + conversationId ??= update.ConversationId; + } + + // Assert + // Should have client function call and result + Assert.Contains(updates, u => u.Contents.Any(c => c is FunctionCallContent fcc && fcc.Name == "ClientTool")); + Assert.Contains(updates, u => u.Contents.Any(c => c is FunctionResultContent frc && frc.CallId == "call_client")); + // Should have final text response + Assert.Contains(updates, u => u.Contents.Any(c => c is TextContent)); + // All updates should have consistent conversation ID + Assert.NotNull(conversationId); + Assert.All(updates, u => Assert.Equal(conversationId, u.ConversationId)); + } + + [Fact] + public async Task GetStreamingResponseAsync_ExtractsThreadIdFromFunctionCall_OnSubsequentTurnsAsync() + { + // Arrange + AIFunction clientTool = AIFunctionFactory.Create(() => "Result", "ClientTool"); + + var handler = new TestDelegatingHandler(); + // First turn: client function call + handler.AddResponse( + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, + new ToolCallStartEvent { ToolCallId = "call_1", ToolCallName = "ClientTool", ParentMessageId = "msg1" }, + new ToolCallArgsEvent { ToolCallId = "call_1", Delta = "{}" }, + new ToolCallEndEvent { ToolCallId = "call_1" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } + ]); + // FunctionInvokingChatClient automatically calls again after function execution + handler.AddResponse( + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run2" }, + new TextMessageStartEvent { MessageId = "msg2", Role = AGUIRoles.Assistant }, + new TextMessageContentEvent { MessageId = "msg2", Delta = "First done" }, + new TextMessageEndEvent { MessageId = "msg2" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run2" } + ]); + // Third turn: user makes another request with conversation history + handler.AddResponse( + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run3" }, + new TextMessageStartEvent { MessageId = "msg3", Role = AGUIRoles.Assistant }, + new TextMessageContentEvent { MessageId = "msg3", Delta = "Second done" }, + new TextMessageEndEvent { MessageId = "msg3" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run3" } + ]); + using HttpClient httpClient = new(handler); + + var chatClient = new AGUIChatClient(httpClient, "http://localhost/agent", null, AGUIJsonSerializerContext.Default.Options); + var options = new ChatOptions { Tools = [clientTool] }; + List messages = [new ChatMessage(ChatRole.User, "Test")]; + + // Act - First turn + List conversation = [.. messages]; + string? conversationId = null; + await foreach (var update in chatClient.GetStreamingResponseAsync(conversation, options)) + { + conversationId ??= update.ConversationId; + // Collect all updates to build the conversation history + foreach (var content in update.Contents) + { + if (content is FunctionCallContent fcc) + { + conversation.Add(new ChatMessage(ChatRole.Assistant, [fcc])); + } + else if (content is FunctionResultContent frc) + { + conversation.Add(new ChatMessage(ChatRole.Tool, [frc])); + } + else if (content is TextContent tc) + { + var existingAssistant = conversation.LastOrDefault(m => m.Role == ChatRole.Assistant && m.Contents.Any(c => c is TextContent)); + if (existingAssistant == null) + { + conversation.Add(new ChatMessage(ChatRole.Assistant, [tc])); + } + } + } + } + + // Act - Second turn with conversation history including function call + // The thread ID should be extracted from the function call in the conversation history + options.ConversationId = conversationId; + List secondTurnUpdates = []; + await foreach (var update in chatClient.GetStreamingResponseAsync(conversation, options)) + { + secondTurnUpdates.Add(update); + } + + // Assert - Second turn should maintain the same conversation ID + Assert.NotNull(conversationId); + Assert.All(secondTurnUpdates, u => Assert.Equal(conversationId, u.ConversationId)); + Assert.Contains(secondTurnUpdates, u => u.Contents.Any(c => c is TextContent)); + } + + [Fact] + public async Task GetStreamingResponseAsync_MaintainsConsistentThreadId_AcrossMultipleTurnsAsync() + { + // Arrange + var handler = new TestDelegatingHandler(); + // Turn 1 + handler.AddResponse( + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, + new TextMessageStartEvent { MessageId = "msg1", Role = AGUIRoles.Assistant }, + new TextMessageContentEvent { MessageId = "msg1", Delta = "Response 1" }, + new TextMessageEndEvent { MessageId = "msg1" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } + ]); + // Turn 2 + handler.AddResponse( + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run2" }, + new TextMessageStartEvent { MessageId = "msg2", Role = AGUIRoles.Assistant }, + new TextMessageContentEvent { MessageId = "msg2", Delta = "Response 2" }, + new TextMessageEndEvent { MessageId = "msg2" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run2" } + ]); + // Turn 3 + handler.AddResponse( + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run3" }, + new TextMessageStartEvent { MessageId = "msg3", Role = AGUIRoles.Assistant }, + new TextMessageContentEvent { MessageId = "msg3", Delta = "Response 3" }, + new TextMessageEndEvent { MessageId = "msg3" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run3" } + ]); + using HttpClient httpClient = new(handler); + + var chatClient = new AGUIChatClient(httpClient, "http://localhost/agent", null, AGUIJsonSerializerContext.Default.Options); + var options = new ChatOptions { ConversationId = "my-conversation" }; + List messages = [new ChatMessage(ChatRole.User, "Test")]; + + // Act - Execute 3 turns + string? conversationId = null; + for (int i = 0; i < 3; i++) + { + await foreach (var update in chatClient.GetStreamingResponseAsync(messages, options)) + { + conversationId ??= update.ConversationId; + Assert.Equal("my-conversation", update.ConversationId); + } + } + + // Assert + Assert.Equal("my-conversation", conversationId); + } + + [Fact] + public async Task GetStreamingResponseAsync_HandlesEmptyThreadId_GracefullyAsync() + { + // Arrange - Server returns empty thread ID + using HttpClient httpClient = this.CreateMockHttpClient( + [ + new RunStartedEvent { ThreadId = string.Empty, RunId = "run1" }, + new TextMessageStartEvent { MessageId = "msg1", Role = AGUIRoles.Assistant }, + new TextMessageContentEvent { MessageId = "msg1", Delta = "Hello" }, + new TextMessageEndEvent { MessageId = "msg1" }, + new RunFinishedEvent { ThreadId = string.Empty, RunId = "run1" } + ]); + + var chatClient = new AGUIChatClient(httpClient, "http://localhost/agent", null, AGUIJsonSerializerContext.Default.Options); + List messages = [new ChatMessage(ChatRole.User, "Test")]; + + // Act + List updates = []; + await foreach (var update in chatClient.GetStreamingResponseAsync(messages, null)) + { + updates.Add(update); + } + + // Assert - Should generate a conversation ID even with empty server thread ID + Assert.NotEmpty(updates); + Assert.All(updates, u => Assert.NotNull(u.ConversationId)); + Assert.All(updates, u => Assert.NotEmpty(u.ConversationId!)); + } + + [Fact] + public async Task GetStreamingResponseAsync_AdaptsToServerThreadIdChange_MidConversationAsync() + { + // Arrange + var handler = new TestDelegatingHandler(); + // First turn: server returns thread-A + handler.AddResponse( + [ + new RunStartedEvent { ThreadId = "thread-A", RunId = "run1" }, + new TextMessageStartEvent { MessageId = "msg1", Role = AGUIRoles.Assistant }, + new TextMessageContentEvent { MessageId = "msg1", Delta = "First" }, + new TextMessageEndEvent { MessageId = "msg1" }, + new RunFinishedEvent { ThreadId = "thread-A", RunId = "run1" } + ]); + // Second turn: provide thread-A but server returns thread-B + handler.AddResponse( + [ + new RunStartedEvent { ThreadId = "thread-B", RunId = "run2" }, + new TextMessageStartEvent { MessageId = "msg2", Role = AGUIRoles.Assistant }, + new TextMessageContentEvent { MessageId = "msg2", Delta = "Second" }, + new TextMessageEndEvent { MessageId = "msg2" }, + new RunFinishedEvent { ThreadId = "thread-B", RunId = "run2" } + ]); + using HttpClient httpClient = new(handler); + + var chatClient = new AGUIChatClient(httpClient, "http://localhost/agent", null, AGUIJsonSerializerContext.Default.Options); + List messages = [new ChatMessage(ChatRole.User, "Test")]; + + // Act - First turn + string? firstConversationId = null; + await foreach (var update in chatClient.GetStreamingResponseAsync(messages, null)) + { + firstConversationId ??= update.ConversationId; + } + + // Second turn - provide the conversation ID from first turn + var options = new ChatOptions { ConversationId = firstConversationId }; + string? secondConversationId = null; + await foreach (var update in chatClient.GetStreamingResponseAsync(messages, options)) + { + secondConversationId ??= update.ConversationId; + } + + // Assert - Should use client-provided conversation ID, not server's changed ID + Assert.Equal("thread-A", firstConversationId); + Assert.Equal("thread-A", secondConversationId); // Client overrides server's thread-B + } + + [Fact] + public async Task GetStreamingResponseAsync_PresentsServerFunctionResults_AsRegularFunctionResultsAsync() + { + // Arrange - Server function (not in client tool set) + using HttpClient httpClient = this.CreateMockHttpClient( + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, + new ToolCallStartEvent { ToolCallId = "call_1", ToolCallName = "ServerTool", ParentMessageId = "msg1" }, + new ToolCallArgsEvent { ToolCallId = "call_1", Delta = "{\"arg\":\"value\"}" }, + new ToolCallEndEvent { ToolCallId = "call_1" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } + ]); + + var chatClient = new AGUIChatClient(httpClient, "http://localhost/agent", null, AGUIJsonSerializerContext.Default.Options); + List messages = [new ChatMessage(ChatRole.User, "Test")]; + + // Act + List updates = []; + await foreach (var update in chatClient.GetStreamingResponseAsync(messages, null)) + { + updates.Add(update); + } + + // Assert - Server function should be presented as FunctionCallContent (unwrapped from ServerFunctionCallContent) + Assert.Contains(updates, u => u.Contents.Any(c => c is FunctionCallContent fcc && fcc.Name == "ServerTool")); + // Verify it's NOT a ServerFunctionCallContent (internal type should be unwrapped) + Assert.All(updates, u => Assert.DoesNotContain(u.Contents, c => c.GetType().Name == "ServerFunctionCallContent")); + } + + [Fact] + public async Task GetStreamingResponseAsync_HandlesMultipleServerFunctions_InSequenceAsync() + { + // Arrange + var handler = new TestDelegatingHandler(); + // Turn 1: Server function 1 + handler.AddResponse( + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, + new ToolCallStartEvent { ToolCallId = "call_1", ToolCallName = "ServerTool1", ParentMessageId = "msg1" }, + new ToolCallArgsEvent { ToolCallId = "call_1", Delta = "{}" }, + new ToolCallEndEvent { ToolCallId = "call_1" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } + ]); + // Turn 2: Server function 2 + handler.AddResponse( + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run2" }, + new ToolCallStartEvent { ToolCallId = "call_2", ToolCallName = "ServerTool2", ParentMessageId = "msg2" }, + new ToolCallArgsEvent { ToolCallId = "call_2", Delta = "{}" }, + new ToolCallEndEvent { ToolCallId = "call_2" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run2" } + ]); + // Turn 3: Final response + handler.AddResponse( + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run3" }, + new TextMessageStartEvent { MessageId = "msg3", Role = AGUIRoles.Assistant }, + new TextMessageContentEvent { MessageId = "msg3", Delta = "Complete" }, + new TextMessageEndEvent { MessageId = "msg3" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run3" } + ]); + using HttpClient httpClient = new(handler); + + var chatClient = new AGUIChatClient(httpClient, "http://localhost/agent", null, AGUIJsonSerializerContext.Default.Options); + var options = new ChatOptions { ConversationId = "conv1" }; + List messages = [new ChatMessage(ChatRole.User, "Test")]; + + // Act - Execute all 3 turns + List allUpdates = []; + for (int i = 0; i < 3; i++) + { + await foreach (var update in chatClient.GetStreamingResponseAsync(messages, options)) + { + allUpdates.Add(update); + } + } + + // Assert + Assert.Contains(allUpdates, u => u.Contents.Any(c => c is FunctionCallContent fcc && fcc.Name == "ServerTool1")); + Assert.Contains(allUpdates, u => u.Contents.Any(c => c is FunctionCallContent fcc && fcc.Name == "ServerTool2")); + Assert.Contains(allUpdates, u => u.Contents.Any(c => c is TextContent)); + Assert.All(allUpdates, u => Assert.Equal("conv1", u.ConversationId)); + } + + [Fact] + public async Task GetStreamingResponseAsync_MaintainsThreadIdConsistency_WithOnlyServerFunctionsAsync() + { + // Arrange - Full conversation with only server functions + var handler = new TestDelegatingHandler(); + // Turn 1: Server function + handler.AddResponse( + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, + new ToolCallStartEvent { ToolCallId = "call_1", ToolCallName = "ServerTool", ParentMessageId = "msg1" }, + new ToolCallArgsEvent { ToolCallId = "call_1", Delta = "{}" }, + new ToolCallEndEvent { ToolCallId = "call_1" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } + ]); + // Turn 2: Final response + handler.AddResponse( + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run2" }, + new TextMessageStartEvent { MessageId = "msg2", Role = AGUIRoles.Assistant }, + new TextMessageContentEvent { MessageId = "msg2", Delta = "Done" }, + new TextMessageEndEvent { MessageId = "msg2" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run2" } + ]); + using HttpClient httpClient = new(handler); + + var chatClient = new AGUIChatClient(httpClient, "http://localhost/agent", null, AGUIJsonSerializerContext.Default.Options); + List messages = [new ChatMessage(ChatRole.User, "Test")]; + + // Act + string? conversationId = null; + List allUpdates = []; + for (int i = 0; i < 2; i++) + { + await foreach (var update in chatClient.GetStreamingResponseAsync(messages, null)) + { + conversationId ??= update.ConversationId; + allUpdates.Add(update); + } + } + + // Assert - Thread ID should be consistent without client function invocations + Assert.NotNull(conversationId); + Assert.All(allUpdates, u => Assert.Equal(conversationId, u.ConversationId)); + Assert.Contains(allUpdates, u => u.Contents.Any(c => c is FunctionCallContent)); + Assert.Contains(allUpdates, u => u.Contents.Any(c => c is TextContent)); + } + + [Fact] + public async Task GetStreamingResponseAsync_StoresConversationIdInAdditionalProperties_WithoutMutatingOptionsAsync() + { + // Arrange + using HttpClient httpClient = this.CreateMockHttpClient( + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, + new TextMessageStartEvent { MessageId = "msg1", Role = AGUIRoles.Assistant }, + new TextMessageContentEvent { MessageId = "msg1", Delta = "Hello" }, + new TextMessageEndEvent { MessageId = "msg1" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } + ]); + + var chatClient = new AGUIChatClient(httpClient, "http://localhost/agent", null, AGUIJsonSerializerContext.Default.Options); + var options = new ChatOptions { ConversationId = "my-conversation-123" }; + var originalConversationId = options.ConversationId; + var originalAdditionalProperties = options.AdditionalProperties; + List messages = [new ChatMessage(ChatRole.User, "Test")]; + + // Act + await foreach (var update in chatClient.GetStreamingResponseAsync(messages, options)) + { + // Just consume the stream + } + + // Assert - Original options should not be mutated + Assert.Equal(originalConversationId, options.ConversationId); + Assert.Equal(originalAdditionalProperties, options.AdditionalProperties); + } + + [Fact] + public async Task GetStreamingResponseAsync_EnsuresConversationIdIsNull_ForInnerClientAsync() + { + // Arrange - Use a custom handler to capture what's sent to the inner layer + var captureHandler = new CapturingTestDelegatingHandler(); + captureHandler.AddResponse( + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, + new TextMessageStartEvent { MessageId = "msg1", Role = AGUIRoles.Assistant }, + new TextMessageContentEvent { MessageId = "msg1", Delta = "Hello" }, + new TextMessageEndEvent { MessageId = "msg1" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } + ]); + using HttpClient httpClient = new(captureHandler); + + var chatClient = new AGUIChatClient(httpClient, "http://localhost/agent", null, AGUIJsonSerializerContext.Default.Options); + var options = new ChatOptions { ConversationId = "my-conversation-123" }; + List messages = [new ChatMessage(ChatRole.User, "Test")]; + + // Act + await foreach (var _ in chatClient.GetStreamingResponseAsync(messages, options)) + { + // Just consume the stream + } + + // Assert - The inner handler should see the full message history being sent + // This is implicitly tested by the fact that all messages are sent in the request + // AG-UI requirement: full history on every turn (which happens when ConversationId is null for FunctionInvokingChatClient) + Assert.True(captureHandler.RequestWasMade); + } + + [Fact] + public async Task GetStreamingResponseAsync_ExtractsStateFromDataContent_AndRemovesStateMessageAsync() + { + // Arrange + var stateData = new { counter = 42, status = "active" }; + string stateJson = JsonSerializer.Serialize(stateData); + byte[] stateBytes = System.Text.Encoding.UTF8.GetBytes(stateJson); + var dataContent = new DataContent(stateBytes, "application/json"); + + var captureHandler = new StateCapturingTestDelegatingHandler(); + captureHandler.AddResponse( + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, + new TextMessageStartEvent { MessageId = "msg1", Role = AGUIRoles.Assistant }, + new TextMessageContentEvent { MessageId = "msg1", Delta = "Response" }, + new TextMessageEndEvent { MessageId = "msg1" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } + ]); + using HttpClient httpClient = new(captureHandler); + + var chatClient = new AGUIChatClient(httpClient, "http://localhost/agent", null, AGUIJsonSerializerContext.Default.Options); + List messages = + [ + new ChatMessage(ChatRole.User, "Hello"), + new ChatMessage(ChatRole.System, [dataContent]) + ]; + + // Act + await foreach (var _ in chatClient.GetStreamingResponseAsync(messages, null)) + { + // Just consume the stream + } + + // Assert + Assert.True(captureHandler.RequestWasMade); + Assert.NotNull(captureHandler.CapturedState); + Assert.Equal(42, captureHandler.CapturedState.Value.GetProperty("counter").GetInt32()); + Assert.Equal("active", captureHandler.CapturedState.Value.GetProperty("status").GetString()); + + // Verify state message was removed - only user message should be in the request + Assert.Equal(1, captureHandler.CapturedMessageCount); + } + + [Fact] + public async Task GetStreamingResponseAsync_WithNoStateDataContent_SendsEmptyStateAsync() + { + // Arrange + var captureHandler = new StateCapturingTestDelegatingHandler(); + captureHandler.AddResponse( + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, + new TextMessageStartEvent { MessageId = "msg1", Role = AGUIRoles.Assistant }, + new TextMessageContentEvent { MessageId = "msg1", Delta = "Response" }, + new TextMessageEndEvent { MessageId = "msg1" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } + ]); + using HttpClient httpClient = new(captureHandler); + + var chatClient = new AGUIChatClient(httpClient, "http://localhost/agent", null, AGUIJsonSerializerContext.Default.Options); + List messages = [new ChatMessage(ChatRole.User, "Hello")]; + + // Act + await foreach (var _ in chatClient.GetStreamingResponseAsync(messages, null)) + { + // Just consume the stream + } + + // Assert + Assert.True(captureHandler.RequestWasMade); + Assert.Null(captureHandler.CapturedState); + } + + [Fact] + public async Task GetStreamingResponseAsync_WithMalformedStateJson_ThrowsInvalidOperationExceptionAsync() + { + // Arrange + byte[] invalidJson = System.Text.Encoding.UTF8.GetBytes("{invalid json"); + var dataContent = new DataContent(invalidJson, "application/json"); + + using HttpClient httpClient = this.CreateMockHttpClient([]); + + var chatClient = new AGUIChatClient(httpClient, "http://localhost/agent", null, AGUIJsonSerializerContext.Default.Options); + List messages = + [ + new ChatMessage(ChatRole.User, "Hello"), + new ChatMessage(ChatRole.System, [dataContent]) + ]; + + // Act & Assert + InvalidOperationException ex = await Assert.ThrowsAsync(async () => + { + await foreach (var _ in chatClient.GetStreamingResponseAsync(messages, null)) + { + // Just consume the stream + } + }); + + Assert.Contains("Failed to deserialize state JSON", ex.Message); + } + + [Fact] + public async Task GetStreamingResponseAsync_WithEmptyStateObject_SendsEmptyObjectAsync() + { + // Arrange + var emptyState = new { }; + string stateJson = JsonSerializer.Serialize(emptyState); + byte[] stateBytes = System.Text.Encoding.UTF8.GetBytes(stateJson); + var dataContent = new DataContent(stateBytes, "application/json"); + + var captureHandler = new StateCapturingTestDelegatingHandler(); + captureHandler.AddResponse( + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } + ]); + using HttpClient httpClient = new(captureHandler); + + var chatClient = new AGUIChatClient(httpClient, "http://localhost/agent", null, AGUIJsonSerializerContext.Default.Options); + List messages = + [ + new ChatMessage(ChatRole.User, "Hello"), + new ChatMessage(ChatRole.System, [dataContent]) + ]; + + // Act + await foreach (var _ in chatClient.GetStreamingResponseAsync(messages, null)) + { + // Just consume the stream + } + + // Assert + Assert.True(captureHandler.RequestWasMade); + Assert.NotNull(captureHandler.CapturedState); + Assert.Equal(JsonValueKind.Object, captureHandler.CapturedState.Value.ValueKind); + } + + [Fact] + public async Task GetStreamingResponseAsync_OnlyProcessesDataContentFromLastMessage_IgnoresEarlierOnesAsync() + { + // Arrange + var oldState = new { counter = 10 }; + string oldStateJson = JsonSerializer.Serialize(oldState); + byte[] oldStateBytes = System.Text.Encoding.UTF8.GetBytes(oldStateJson); + var oldDataContent = new DataContent(oldStateBytes, "application/json"); + + var newState = new { counter = 20 }; + string newStateJson = JsonSerializer.Serialize(newState); + byte[] newStateBytes = System.Text.Encoding.UTF8.GetBytes(newStateJson); + var newDataContent = new DataContent(newStateBytes, "application/json"); + + var captureHandler = new StateCapturingTestDelegatingHandler(); + captureHandler.AddResponse( + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } + ]); + using HttpClient httpClient = new(captureHandler); + + var chatClient = new AGUIChatClient(httpClient, "http://localhost/agent", null, AGUIJsonSerializerContext.Default.Options); + List messages = + [ + new ChatMessage(ChatRole.User, "First message"), + new ChatMessage(ChatRole.System, [oldDataContent]), + new ChatMessage(ChatRole.User, "Second message"), + new ChatMessage(ChatRole.System, [newDataContent]) + ]; + + // Act + await foreach (var _ in chatClient.GetStreamingResponseAsync(messages, null)) + { + // Just consume the stream + } + + // Assert + Assert.True(captureHandler.RequestWasMade); + Assert.NotNull(captureHandler.CapturedState); + // Should use the new state from the last message + Assert.Equal(20, captureHandler.CapturedState.Value.GetProperty("counter").GetInt32()); + + // Should have removed only the last state message + Assert.Equal(3, captureHandler.CapturedMessageCount); + } + + [Fact] + public async Task GetStreamingResponseAsync_WithNonJsonMediaType_IgnoresDataContentAsync() + { + // Arrange + byte[] imageData = System.Text.Encoding.UTF8.GetBytes("fake image data"); + var dataContent = new DataContent(imageData, "image/png"); + + var captureHandler = new StateCapturingTestDelegatingHandler(); + captureHandler.AddResponse( + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } + ]); + using HttpClient httpClient = new(captureHandler); + + var chatClient = new AGUIChatClient(httpClient, "http://localhost/agent", null, AGUIJsonSerializerContext.Default.Options); + List messages = + [ + new ChatMessage(ChatRole.User, [new TextContent("Hello"), dataContent]) + ]; + + // Act + await foreach (var _ in chatClient.GetStreamingResponseAsync(messages, null)) + { + // Just consume the stream + } + + // Assert + Assert.True(captureHandler.RequestWasMade); + Assert.Null(captureHandler.CapturedState); + // Message should not be removed since it's not state + Assert.Equal(1, captureHandler.CapturedMessageCount); + } + + [Fact] + public async Task GetStreamingResponseAsync_RoundTripState_PreservesJsonStructureAsync() + { + // Arrange - Server returns state snapshot + var returnedState = new { counter = 100, nested = new { value = "test" } }; + JsonElement stateSnapshot = JsonSerializer.SerializeToElement(returnedState); + + var captureHandler = new StateCapturingTestDelegatingHandler(); + captureHandler.AddResponse( + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, + new StateSnapshotEvent { Snapshot = stateSnapshot }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } + ]); + captureHandler.AddResponse( + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run2" }, + new TextMessageStartEvent { MessageId = "msg1", Role = AGUIRoles.Assistant }, + new TextMessageContentEvent { MessageId = "msg1", Delta = "Done" }, + new TextMessageEndEvent { MessageId = "msg1" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run2" } + ]); + using HttpClient httpClient = new(captureHandler); + + var chatClient = new AGUIChatClient(httpClient, "http://localhost/agent", null, AGUIJsonSerializerContext.Default.Options); + List messages = [new ChatMessage(ChatRole.User, "Hello")]; + + // Act - First turn: receive state + DataContent? receivedStateContent = null; + await foreach (var update in chatClient.GetStreamingResponseAsync(messages, null)) + { + if (update.Contents.Any(c => c is DataContent dc && dc.MediaType == "application/json")) + { + receivedStateContent = (DataContent)update.Contents.First(c => c is DataContent); + } + } + + // Second turn: send the received state back + Assert.NotNull(receivedStateContent); + messages.Add(new ChatMessage(ChatRole.System, [receivedStateContent])); + await foreach (var _ in chatClient.GetStreamingResponseAsync(messages, null)) + { + // Just consume the stream + } + + // Assert - Verify the round-tripped state + Assert.NotNull(captureHandler.CapturedState); + Assert.Equal(100, captureHandler.CapturedState.Value.GetProperty("counter").GetInt32()); + Assert.Equal("test", captureHandler.CapturedState.Value.GetProperty("nested").GetProperty("value").GetString()); + } + + [Fact] + public async Task GetStreamingResponseAsync_ReceivesStateSnapshot_AsDataContentWithAdditionalPropertiesAsync() + { + // Arrange + var state = new { sessionId = "abc123", step = 5 }; + JsonElement stateSnapshot = JsonSerializer.SerializeToElement(state); + + using HttpClient httpClient = this.CreateMockHttpClient( + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, + new StateSnapshotEvent { Snapshot = stateSnapshot }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } + ]); + + var chatClient = new AGUIChatClient(httpClient, "http://localhost/agent", null, AGUIJsonSerializerContext.Default.Options); + List messages = [new ChatMessage(ChatRole.User, "Test")]; + + // Act + List updates = []; + await foreach (var update in chatClient.GetStreamingResponseAsync(messages, null)) + { + updates.Add(update); + } + + // Assert + ChatResponseUpdate stateUpdate = updates.First(u => u.Contents.Any(c => c is DataContent)); + Assert.NotNull(stateUpdate.AdditionalProperties); + Assert.True((bool)stateUpdate.AdditionalProperties!["is_state_snapshot"]!); + + DataContent dataContent = (DataContent)stateUpdate.Contents[0]; + Assert.Equal("application/json", dataContent.MediaType); + + string jsonText = System.Text.Encoding.UTF8.GetString(dataContent.Data.ToArray()); + JsonElement deserializedState = JsonElement.Parse(jsonText); + Assert.Equal("abc123", deserializedState.GetProperty("sessionId").GetString()); + Assert.Equal(5, deserializedState.GetProperty("step").GetInt32()); + } +} + +internal sealed class TestDelegatingHandler : DelegatingHandler +{ + private readonly Queue>> _responseFactories = new(); + private readonly List _capturedRunIds = []; + + public IReadOnlyList CapturedRunIds => this._capturedRunIds; + + public void AddResponse(BaseEvent[] events) + { + this._responseFactories.Enqueue(_ => Task.FromResult(CreateResponse(events))); + } + + public void AddResponseWithCapture(BaseEvent[] events) + { + this._responseFactories.Enqueue(async request => + { + await this.CaptureRunIdAsync(request); + return CreateResponse(events); + }); + } + + protected override async Task SendAsync(HttpRequestMessage request, CancellationToken cancellationToken) + { + if (this._responseFactories.Count == 0) + { + // Log request count for debugging + throw new InvalidOperationException($"No more responses configured for TestDelegatingHandler. Total requests made: {this._capturedRunIds.Count}"); + } + + var factory = this._responseFactories.Dequeue(); + return await factory(request); + } + + private static HttpResponseMessage CreateResponse(BaseEvent[] events) + { + string sseContent = string.Join("", events.Select(e => + $"data: {JsonSerializer.Serialize(e, AGUIJsonSerializerContext.Default.BaseEvent)}\n\n")); + + return new HttpResponseMessage + { + StatusCode = HttpStatusCode.OK, + Content = new StringContent(sseContent) + }; + } + + private async Task CaptureRunIdAsync(HttpRequestMessage request) + { + string requestBody = await request.Content!.ReadAsStringAsync().ConfigureAwait(false); + RunAgentInput? input = JsonSerializer.Deserialize(requestBody, AGUIJsonSerializerContext.Default.RunAgentInput); + if (input != null) + { + this._capturedRunIds.Add(input.RunId); + } + } +} + +internal sealed class CapturingTestDelegatingHandler : DelegatingHandler +{ + private readonly Queue>> _responseFactories = new(); + + public bool RequestWasMade { get; private set; } + + public void AddResponse(BaseEvent[] events) + { + this._responseFactories.Enqueue(_ => Task.FromResult(CreateResponse(events))); + } + + protected override async Task SendAsync(HttpRequestMessage request, CancellationToken cancellationToken) + { + this.RequestWasMade = true; + + if (this._responseFactories.Count == 0) + { + throw new InvalidOperationException("No more responses configured for CapturingTestDelegatingHandler."); + } + + var factory = this._responseFactories.Dequeue(); + return await factory(request); + } + + private static HttpResponseMessage CreateResponse(BaseEvent[] events) + { + string sseContent = string.Join("", events.Select(e => + $"data: {JsonSerializer.Serialize(e, AGUIJsonSerializerContext.Default.BaseEvent)}\n\n")); + + return new HttpResponseMessage + { + StatusCode = HttpStatusCode.OK, + Content = new StringContent(sseContent) + }; + } +} + +internal sealed class StateCapturingTestDelegatingHandler : DelegatingHandler +{ + private readonly Queue>> _responseFactories = new(); + + public bool RequestWasMade { get; private set; } + public JsonElement? CapturedState { get; private set; } + public int CapturedMessageCount { get; private set; } + + public void AddResponse(BaseEvent[] events) + { + this._responseFactories.Enqueue(_ => Task.FromResult(CreateResponse(events))); + } + + protected override async Task SendAsync(HttpRequestMessage request, CancellationToken cancellationToken) + { + this.RequestWasMade = true; + + // Capture the state and message count from the request +#if !NET + string requestBody = await request.Content!.ReadAsStringAsync().ConfigureAwait(false); +#else + string requestBody = await request.Content!.ReadAsStringAsync(cancellationToken).ConfigureAwait(false); +#endif + RunAgentInput? input = JsonSerializer.Deserialize(requestBody, AGUIJsonSerializerContext.Default.RunAgentInput); + if (input != null) + { + if (input.State.ValueKind is not JsonValueKind.Undefined and not JsonValueKind.Null) + { + this.CapturedState = input.State; + } + this.CapturedMessageCount = input.Messages.Count(); + } + + if (this._responseFactories.Count == 0) + { + throw new InvalidOperationException("No more responses configured for StateCapturingTestDelegatingHandler."); + } + + var factory = this._responseFactories.Dequeue(); + return await factory(request); + } + + private static HttpResponseMessage CreateResponse(BaseEvent[] events) + { + string sseContent = string.Join("", events.Select(e => + $"data: {JsonSerializer.Serialize(e, AGUIJsonSerializerContext.Default.BaseEvent)}\n\n")); + + return new HttpResponseMessage + { + StatusCode = HttpStatusCode.OK, + Content = new StringContent(sseContent) + }; + } +} diff --git a/dotnet/tests/Microsoft.Agents.AI.AGUI.UnitTests/AGUIChatMessageExtensionsTests.cs b/dotnet/tests/Microsoft.Agents.AI.AGUI.UnitTests/AGUIChatMessageExtensionsTests.cs index d57cac1990..bc3a73fb4c 100644 --- a/dotnet/tests/Microsoft.Agents.AI.AGUI.UnitTests/AGUIChatMessageExtensionsTests.cs +++ b/dotnet/tests/Microsoft.Agents.AI.AGUI.UnitTests/AGUIChatMessageExtensionsTests.cs @@ -3,11 +3,34 @@ using System; using System.Collections.Generic; using System.Linq; +using System.Text.Json.Serialization; using Microsoft.Agents.AI.AGUI.Shared; using Microsoft.Extensions.AI; namespace Microsoft.Agents.AI.AGUI.UnitTests; +// Custom complex type for testing tool call parameters +public sealed class WeatherRequest +{ + public string Location { get; set; } = string.Empty; + public string Units { get; set; } = "celsius"; + public bool IncludeForecast { get; set; } +} + +// Custom complex type for testing tool call results +public sealed class WeatherResponse +{ + public double Temperature { get; set; } + public string Conditions { get; set; } = string.Empty; + public DateTime Timestamp { get; set; } +} + +// Custom JsonSerializerContext for the custom types +[JsonSerializable(typeof(WeatherRequest))] +[JsonSerializable(typeof(WeatherResponse))] +[JsonSerializable(typeof(Dictionary))] +internal sealed partial class CustomTypesContext : JsonSerializerContext; + /// /// Unit tests for the class. /// @@ -20,7 +43,7 @@ public void AsChatMessages_WithEmptyCollection_ReturnsEmptyList() List aguiMessages = []; // Act - IEnumerable chatMessages = aguiMessages.AsChatMessages(); + IEnumerable chatMessages = aguiMessages.AsChatMessages(AGUIJsonSerializerContext.Default.Options); // Assert Assert.NotNull(chatMessages); @@ -33,16 +56,15 @@ public void AsChatMessages_WithSingleMessage_ConvertsToChatMessageCorrectly() // Arrange List aguiMessages = [ - new AGUIMessage + new AGUIUserMessage { Id = "msg1", - Role = AGUIRoles.User, Content = "Hello" } ]; // Act - IEnumerable chatMessages = aguiMessages.AsChatMessages(); + IEnumerable chatMessages = aguiMessages.AsChatMessages(AGUIJsonSerializerContext.Default.Options); // Assert ChatMessage message = Assert.Single(chatMessages); @@ -56,13 +78,13 @@ public void AsChatMessages_WithMultipleMessages_PreservesOrder() // Arrange List aguiMessages = [ - new AGUIMessage { Id = "msg1", Role = AGUIRoles.User, Content = "First" }, - new AGUIMessage { Id = "msg2", Role = AGUIRoles.Assistant, Content = "Second" }, - new AGUIMessage { Id = "msg3", Role = AGUIRoles.User, Content = "Third" } + new AGUIUserMessage { Id = "msg1", Content = "First" }, + new AGUIAssistantMessage { Id = "msg2", Content = "Second" }, + new AGUIUserMessage { Id = "msg3", Content = "Third" } ]; // Act - List chatMessages = aguiMessages.AsChatMessages().ToList(); + List chatMessages = aguiMessages.AsChatMessages(AGUIJsonSerializerContext.Default.Options).ToList(); // Assert Assert.Equal(3, chatMessages.Count); @@ -77,14 +99,14 @@ public void AsChatMessages_MapsAllSupportedRoleTypes_Correctly() // Arrange List aguiMessages = [ - new AGUIMessage { Id = "msg1", Role = AGUIRoles.System, Content = "System message" }, - new AGUIMessage { Id = "msg2", Role = AGUIRoles.User, Content = "User message" }, - new AGUIMessage { Id = "msg3", Role = AGUIRoles.Assistant, Content = "Assistant message" }, - new AGUIMessage { Id = "msg4", Role = AGUIRoles.Developer, Content = "Developer message" } + new AGUISystemMessage { Id = "msg1", Content = "System message" }, + new AGUIUserMessage { Id = "msg2", Content = "User message" }, + new AGUIAssistantMessage { Id = "msg3", Content = "Assistant message" }, + new AGUIDeveloperMessage { Id = "msg4", Content = "Developer message" } ]; // Act - List chatMessages = aguiMessages.AsChatMessages().ToList(); + List chatMessages = aguiMessages.AsChatMessages(AGUIJsonSerializerContext.Default.Options).ToList(); // Assert Assert.Equal(4, chatMessages.Count); @@ -101,7 +123,7 @@ public void AsAGUIMessages_WithEmptyCollection_ReturnsEmptyList() List chatMessages = []; // Act - IEnumerable aguiMessages = chatMessages.AsAGUIMessages(); + IEnumerable aguiMessages = chatMessages.AsAGUIMessages(AGUIJsonSerializerContext.Default.Options); // Assert Assert.NotNull(aguiMessages); @@ -118,13 +140,13 @@ public void AsAGUIMessages_WithSingleMessage_ConvertsToAGUIMessageCorrectly() ]; // Act - IEnumerable aguiMessages = chatMessages.AsAGUIMessages(); + IEnumerable aguiMessages = chatMessages.AsAGUIMessages(AGUIJsonSerializerContext.Default.Options); // Assert AGUIMessage message = Assert.Single(aguiMessages); Assert.Equal("msg1", message.Id); Assert.Equal(AGUIRoles.User, message.Role); - Assert.Equal("Hello", message.Content); + Assert.Equal("Hello", ((AGUIUserMessage)message).Content); } [Fact] @@ -139,13 +161,13 @@ public void AsAGUIMessages_WithMultipleMessages_PreservesOrder() ]; // Act - List aguiMessages = chatMessages.AsAGUIMessages().ToList(); + List aguiMessages = chatMessages.AsAGUIMessages(AGUIJsonSerializerContext.Default.Options).ToList(); // Assert Assert.Equal(3, aguiMessages.Count); - Assert.Equal("First", aguiMessages[0].Content); - Assert.Equal("Second", aguiMessages[1].Content); - Assert.Equal("Third", aguiMessages[2].Content); + Assert.Equal("First", ((AGUIUserMessage)aguiMessages[0]).Content); + Assert.Equal("Second", ((AGUIAssistantMessage)aguiMessages[1]).Content); + Assert.Equal("Third", ((AGUIUserMessage)aguiMessages[2]).Content); } [Fact] @@ -158,7 +180,7 @@ public void AsAGUIMessages_PreservesMessageId_WhenPresent() ]; // Act - IEnumerable aguiMessages = chatMessages.AsAGUIMessages(); + IEnumerable aguiMessages = chatMessages.AsAGUIMessages(AGUIJsonSerializerContext.Default.Options); // Assert AGUIMessage message = Assert.Single(aguiMessages); @@ -185,4 +207,438 @@ public void MapChatRole_WithUnknownRole_ThrowsInvalidOperationException() // Arrange & Act & Assert Assert.Throws(() => AGUIChatMessageExtensions.MapChatRole("unknown")); } + + [Fact] + public void AsAGUIMessages_WithToolResultMessage_SerializesResultCorrectly() + { + // Arrange + var result = new Dictionary { ["temperature"] = 72, ["condition"] = "Sunny" }; + FunctionResultContent toolResult = new("call_123", result); + ChatMessage toolMessage = new(ChatRole.Tool, [toolResult]); + List messages = [toolMessage]; + + // Act + List aguiMessages = messages.AsAGUIMessages(AGUIJsonSerializerContext.Default.Options).ToList(); + + // Assert + AGUIMessage aguiMessage = Assert.Single(aguiMessages); + Assert.Equal(AGUIRoles.Tool, aguiMessage.Role); + Assert.Equal("call_123", ((AGUIToolMessage)aguiMessage).ToolCallId); + Assert.NotEmpty(((AGUIToolMessage)aguiMessage).Content); + // Content should be serialized JSON + Assert.Contains("temperature", ((AGUIToolMessage)aguiMessage).Content); + Assert.Contains("72", ((AGUIToolMessage)aguiMessage).Content); + } + + [Fact] + public void AsAGUIMessages_WithNullToolResult_HandlesGracefully() + { + // Arrange + FunctionResultContent toolResult = new("call_456", null); + ChatMessage toolMessage = new(ChatRole.Tool, [toolResult]); + List messages = [toolMessage]; + + // Act + List aguiMessages = messages.AsAGUIMessages(AGUIJsonSerializerContext.Default.Options).ToList(); + + // Assert + AGUIMessage aguiMessage = Assert.Single(aguiMessages); + Assert.Equal(AGUIRoles.Tool, aguiMessage.Role); + Assert.Equal("call_456", ((AGUIToolMessage)aguiMessage).ToolCallId); + Assert.Equal(string.Empty, ((AGUIToolMessage)aguiMessage).Content); + } + + [Fact] + public void AsAGUIMessages_WithoutTypeInfoResolver_ThrowsInvalidOperationException() + { + // Arrange + FunctionResultContent toolResult = new("call_789", "Result"); + ChatMessage toolMessage = new(ChatRole.Tool, [toolResult]); + List messages = [toolMessage]; + System.Text.Json.JsonSerializerOptions optionsWithoutResolver = new(); + + // Act & Assert + NotSupportedException ex = Assert.Throws(() => messages.AsAGUIMessages(optionsWithoutResolver).ToList()); + Assert.Contains("JsonTypeInfo", ex.Message); + } + + [Fact] + public void AsChatMessages_WithToolMessage_DeserializesResultCorrectly() + { + // Arrange + const string JsonContent = "{\"status\":\"success\",\"value\":42}"; + List aguiMessages = + [ + new AGUIToolMessage + { + Id = "msg1", + Content = JsonContent, + ToolCallId = "call_abc" + } + ]; + + // Act + List chatMessages = aguiMessages.AsChatMessages(AGUIJsonSerializerContext.Default.Options).ToList(); + + // Assert + ChatMessage message = Assert.Single(chatMessages); + Assert.Equal(ChatRole.Tool, message.Role); + FunctionResultContent result = Assert.IsType(message.Contents[0]); + Assert.Equal("call_abc", result.CallId); + Assert.NotNull(result.Result); + } + + [Fact] + public void AsChatMessages_WithEmptyToolContent_CreatesNullResult() + { + // Arrange + List aguiMessages = + [ + new AGUIToolMessage + { + Id = "msg1", + Content = string.Empty, + ToolCallId = "call_def" + } + ]; + + // Act + List chatMessages = aguiMessages.AsChatMessages(AGUIJsonSerializerContext.Default.Options).ToList(); + + // Assert + ChatMessage message = Assert.Single(chatMessages); + FunctionResultContent result = Assert.IsType(message.Contents[0]); + Assert.Equal("call_def", result.CallId); + Assert.Equal(string.Empty, result.Result); + } + + [Fact] + public void AsChatMessages_WithToolMessageWithoutCallId_TreatsAsRegularMessage() + { + // Arrange - use valid JSON for Content + List aguiMessages = + [ + new AGUIToolMessage + { + Id = "msg1", + Content = "{\"result\":\"Some content\"}", + ToolCallId = string.Empty + } + ]; + + // Act + List chatMessages = aguiMessages.AsChatMessages(AGUIJsonSerializerContext.Default.Options).ToList(); + + // Assert + ChatMessage message = Assert.Single(chatMessages); + Assert.Equal(ChatRole.Tool, message.Role); + var resultContent = Assert.IsType(message.Contents.First()); + Assert.Equal(string.Empty, resultContent.CallId); + } + + [Fact] + public void RoundTrip_ToolResultMessage_PreservesData() + { + // Arrange + var resultData = new Dictionary { ["location"] = "Seattle", ["temperature"] = 68, ["forecast"] = "Partly cloudy" }; + FunctionResultContent originalResult = new("call_roundtrip", resultData); + ChatMessage originalMessage = new(ChatRole.Tool, [originalResult]); + + // Act - Convert to AGUI and back + List originalList = [originalMessage]; + AGUIMessage aguiMessage = originalList.AsAGUIMessages(AGUIJsonSerializerContext.Default.Options).Single(); + List aguiList = [aguiMessage]; + ChatMessage reconstructedMessage = aguiList.AsChatMessages(AGUIJsonSerializerContext.Default.Options).Single(); + + // Assert + Assert.Equal(ChatRole.Tool, reconstructedMessage.Role); + FunctionResultContent reconstructedResult = Assert.IsType(reconstructedMessage.Contents[0]); + Assert.Equal("call_roundtrip", reconstructedResult.CallId); + Assert.NotNull(reconstructedResult.Result); + } + + [Fact] + public void MapChatRole_WithToolRole_ReturnsToolChatRole() + { + // Arrange & Act + ChatRole role = AGUIChatMessageExtensions.MapChatRole(AGUIRoles.Tool); + + // Assert + Assert.Equal(ChatRole.Tool, role); + } + + #region Custom Type Serialization Tests + + [Fact] + public void AsChatMessages_WithFunctionCallContainingCustomType_SerializesCorrectly() + { + // Arrange + var customRequest = new WeatherRequest { Location = "Seattle", Units = "fahrenheit", IncludeForecast = true }; + var parameters = new Dictionary + { + ["location"] = customRequest.Location, + ["units"] = customRequest.Units, + ["includeForecast"] = customRequest.IncludeForecast + }; + + List aguiMessages = + [ + new AGUIAssistantMessage + { + Id = "msg1", + ToolCalls = + [ + new AGUIToolCall + { + Id = "call_1", + Function = new AGUIFunctionCall + { + Name = "GetWeather", + Arguments = System.Text.Json.JsonSerializer.Serialize(parameters, AGUIJsonSerializerContext.Default.Options) + } + } + ] + } + ]; + + // Combine contexts for serialization + var combinedOptions = new System.Text.Json.JsonSerializerOptions + { + TypeInfoResolver = System.Text.Json.Serialization.Metadata.JsonTypeInfoResolver.Combine( + AGUIJsonSerializerContext.Default, + CustomTypesContext.Default) + }; + + // Act + IEnumerable chatMessages = aguiMessages.AsChatMessages(combinedOptions); + + // Assert + ChatMessage message = Assert.Single(chatMessages); + Assert.Equal(ChatRole.Assistant, message.Role); + var toolCallContent = Assert.IsType(message.Contents.First()); + Assert.Equal("call_1", toolCallContent.CallId); + Assert.Equal("GetWeather", toolCallContent.Name); + Assert.NotNull(toolCallContent.Arguments); + // Compare as strings since deserialization produces JsonElement objects + Assert.Equal("Seattle", ((System.Text.Json.JsonElement)toolCallContent.Arguments["location"]!).GetString()); + Assert.Equal("fahrenheit", ((System.Text.Json.JsonElement)toolCallContent.Arguments["units"]!).GetString()); + Assert.True(toolCallContent.Arguments["includeForecast"] is System.Text.Json.JsonElement j && j.GetBoolean()); + } + + [Fact] + public void AsAGUIMessages_WithFunctionResultContainingCustomType_SerializesCorrectly() + { + // Arrange + var customResponse = new WeatherResponse { Temperature = 72.5, Conditions = "Sunny", Timestamp = DateTime.UtcNow }; + var resultObject = new Dictionary + { + ["temperature"] = customResponse.Temperature, + ["conditions"] = customResponse.Conditions, + ["timestamp"] = customResponse.Timestamp.ToString("O") + }; + + var resultJson = System.Text.Json.JsonSerializer.Serialize(resultObject, AGUIJsonSerializerContext.Default.Options); + var functionResult = new FunctionResultContent("call_1", System.Text.Json.JsonSerializer.Deserialize(resultJson, AGUIJsonSerializerContext.Default.Options)); + List chatMessages = + [ + new ChatMessage(ChatRole.Tool, [functionResult]) + ]; + + // Combine contexts for serialization + var combinedOptions = new System.Text.Json.JsonSerializerOptions + { + TypeInfoResolver = System.Text.Json.Serialization.Metadata.JsonTypeInfoResolver.Combine( + AGUIJsonSerializerContext.Default, + CustomTypesContext.Default) + }; + + // Act + IEnumerable aguiMessages = chatMessages.AsAGUIMessages(combinedOptions); + + // Assert + AGUIMessage message = Assert.Single(aguiMessages); + var toolMessage = Assert.IsType(message); + Assert.Equal("call_1", toolMessage.ToolCallId); + Assert.NotNull(toolMessage.Content); + + // Verify the content can be deserialized back + var deserializedResult = System.Text.Json.JsonSerializer.Deserialize>( + toolMessage.Content, + combinedOptions); + Assert.NotNull(deserializedResult); + Assert.Equal(72.5, deserializedResult["temperature"].GetDouble()); + Assert.Equal("Sunny", deserializedResult["conditions"].GetString()); + } + + [Fact] + public void RoundTrip_WithCustomTypesInFunctionCallAndResult_PreservesData() + { + // Arrange + var customRequest = new WeatherRequest { Location = "New York", Units = "celsius", IncludeForecast = false }; + var parameters = new Dictionary + { + ["location"] = customRequest.Location, + ["units"] = customRequest.Units, + ["includeForecast"] = customRequest.IncludeForecast + }; + + var customResponse = new WeatherResponse { Temperature = 22.3, Conditions = "Cloudy", Timestamp = DateTime.UtcNow }; + var resultObject = new Dictionary + { + ["temperature"] = customResponse.Temperature, + ["conditions"] = customResponse.Conditions, + ["timestamp"] = customResponse.Timestamp.ToString("O") + }; + + var resultJson = System.Text.Json.JsonSerializer.Serialize(resultObject, AGUIJsonSerializerContext.Default.Options); + var resultElement = System.Text.Json.JsonSerializer.Deserialize(resultJson, AGUIJsonSerializerContext.Default.Options); + + List originalChatMessages = + [ + new ChatMessage(ChatRole.Assistant, [new FunctionCallContent("call_1", "GetWeather", parameters)]), + new ChatMessage(ChatRole.Tool, [new FunctionResultContent("call_1", resultElement)]) + ]; + + // Combine contexts for serialization + var combinedOptions = new System.Text.Json.JsonSerializerOptions + { + TypeInfoResolver = System.Text.Json.Serialization.Metadata.JsonTypeInfoResolver.Combine( + AGUIJsonSerializerContext.Default, + CustomTypesContext.Default) + }; + + // Act - Convert to AGUI messages and back + IEnumerable aguiMessages = originalChatMessages.AsAGUIMessages(combinedOptions); + List roundTrippedChatMessages = aguiMessages.AsChatMessages(combinedOptions).ToList(); + + // Assert + Assert.Equal(2, roundTrippedChatMessages.Count); + + // Verify function call + ChatMessage callMessage = roundTrippedChatMessages[0]; + Assert.Equal(ChatRole.Assistant, callMessage.Role); + var functionCall = Assert.IsType(callMessage.Contents.First()); + Assert.Equal("call_1", functionCall.CallId); + Assert.Equal("GetWeather", functionCall.Name); + Assert.NotNull(functionCall.Arguments); + // Compare string values from JsonElement + Assert.Equal(customRequest.Location, functionCall.Arguments["location"]?.ToString()); + Assert.Equal(customRequest.Units, functionCall.Arguments["units"]?.ToString()); + + // Verify function result + ChatMessage resultMessage = roundTrippedChatMessages[1]; + Assert.Equal(ChatRole.Tool, resultMessage.Role); + var functionResultContent = Assert.IsType(resultMessage.Contents.First()); + Assert.Equal("call_1", functionResultContent.CallId); + Assert.NotNull(functionResultContent.Result); + } + + [Fact] + public void AsAGUIMessages_WithNestedCustomObjects_HandlesComplexSerialization() + { + // Arrange - nested custom types + var nestedParameters = new Dictionary + { + ["request"] = new Dictionary + { + ["location"] = "Boston", + ["options"] = new Dictionary + { + ["units"] = "fahrenheit", + ["includeHumidity"] = true, + ["daysAhead"] = 5 + } + } + }; + + var functionCall = new FunctionCallContent("call_nested", "GetDetailedWeather", nestedParameters); + List chatMessages = + [ + new ChatMessage(ChatRole.Assistant, [functionCall]) + ]; + + // Combine contexts for serialization + var combinedOptions = new System.Text.Json.JsonSerializerOptions + { + TypeInfoResolver = System.Text.Json.Serialization.Metadata.JsonTypeInfoResolver.Combine( + AGUIJsonSerializerContext.Default, + CustomTypesContext.Default) + }; + + // Act + IEnumerable aguiMessages = chatMessages.AsAGUIMessages(combinedOptions); + + // Assert + AGUIMessage message = Assert.Single(aguiMessages); + var assistantMessage = Assert.IsType(message); + Assert.NotNull(assistantMessage.ToolCalls); + var toolCall = Assert.Single(assistantMessage.ToolCalls); + Assert.Equal("call_nested", toolCall.Id); + Assert.Equal("GetDetailedWeather", toolCall.Function?.Name); + + // Verify nested structure is preserved + var deserializedArgs = System.Text.Json.JsonSerializer.Deserialize>( + toolCall.Function?.Arguments ?? "{}", + combinedOptions); + Assert.NotNull(deserializedArgs); + Assert.True(deserializedArgs.ContainsKey("request")); + } + + [Fact] + public void AsAGUIMessages_WithDictionaryContainingCustomTypes_SerializesDirectly() + { + // Arrange - Create a dictionary with custom type values (not flattened) + var customRequest = new WeatherRequest { Location = "Tokyo", Units = "celsius", IncludeForecast = true }; + var parameters = new Dictionary + { + ["customRequest"] = customRequest, // Custom type as value + ["simpleString"] = "test", + ["simpleNumber"] = 42 + }; + + List chatMessages = + [ + new ChatMessage(ChatRole.Assistant, [new FunctionCallContent("call_custom", "ProcessWeather", parameters)]) + ]; + + // Combine contexts for serialization + var combinedOptions = new System.Text.Json.JsonSerializerOptions + { + TypeInfoResolver = System.Text.Json.Serialization.Metadata.JsonTypeInfoResolver.Combine( + AGUIJsonSerializerContext.Default, + CustomTypesContext.Default) + }; + + // Act + IEnumerable aguiMessages = chatMessages.AsAGUIMessages(combinedOptions); + + // Assert + AGUIMessage message = Assert.Single(aguiMessages); + var assistantMessage = Assert.IsType(message); + Assert.NotNull(assistantMessage.ToolCalls); + var toolCall = Assert.Single(assistantMessage.ToolCalls); + Assert.Equal("call_custom", toolCall.Id); + Assert.Equal("ProcessWeather", toolCall.Function?.Name); + + // Verify custom type was serialized correctly without flattening + var deserializedArgs = System.Text.Json.JsonSerializer.Deserialize>( + toolCall.Function?.Arguments ?? "{}", + combinedOptions); + Assert.NotNull(deserializedArgs); + Assert.True(deserializedArgs.ContainsKey("customRequest")); + Assert.True(deserializedArgs.ContainsKey("simpleString")); + Assert.True(deserializedArgs.ContainsKey("simpleNumber")); + + // Verify the custom type properties are accessible + var customRequestElement = deserializedArgs["customRequest"]; + Assert.Equal("Tokyo", customRequestElement.GetProperty("Location").GetString()); + Assert.Equal("celsius", customRequestElement.GetProperty("Units").GetString()); + Assert.True(customRequestElement.GetProperty("IncludeForecast").GetBoolean()); + + // Verify simple types + Assert.Equal("test", deserializedArgs["simpleString"].GetString()); + Assert.Equal(42, deserializedArgs["simpleNumber"].GetInt32()); + } + + #endregion } diff --git a/dotnet/tests/Microsoft.Agents.AI.AGUI.UnitTests/AGUIHttpServiceTests.cs b/dotnet/tests/Microsoft.Agents.AI.AGUI.UnitTests/AGUIHttpServiceTests.cs index fb40dc622e..b06913c837 100644 --- a/dotnet/tests/Microsoft.Agents.AI.AGUI.UnitTests/AGUIHttpServiceTests.cs +++ b/dotnet/tests/Microsoft.Agents.AI.AGUI.UnitTests/AGUIHttpServiceTests.cs @@ -22,22 +22,22 @@ public sealed class AGUIHttpServiceTests public async Task PostRunAsync_SendsRequestAndParsesSSEStream_SuccessfullyAsync() { // Arrange - BaseEvent[] events = new BaseEvent[] - { + BaseEvent[] events = + [ new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, new TextMessageStartEvent { MessageId = "msg1", Role = AGUIRoles.Assistant }, new TextMessageContentEvent { MessageId = "msg1", Delta = "Hello" }, new TextMessageEndEvent { MessageId = "msg1" }, new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } - }; + ]; - HttpClient httpClient = this.CreateMockHttpClient(events, HttpStatusCode.OK); + HttpClient httpClient = CreateMockHttpClient(events, HttpStatusCode.OK); AGUIHttpService service = new(httpClient, "http://localhost/agent"); RunAgentInput input = new() { ThreadId = "thread1", RunId = "run1", - Messages = [new AGUIMessage { Id = "m1", Role = AGUIRoles.User, Content = "Test" }] + Messages = [new AGUIUserMessage { Id = "m1", Content = "Test" }] }; // Act @@ -60,13 +60,13 @@ public async Task PostRunAsync_SendsRequestAndParsesSSEStream_SuccessfullyAsync( public async Task PostRunAsync_WithNonSuccessStatusCode_ThrowsHttpRequestExceptionAsync() { // Arrange - HttpClient httpClient = this.CreateMockHttpClient([], HttpStatusCode.InternalServerError); + HttpClient httpClient = CreateMockHttpClient([], HttpStatusCode.InternalServerError); AGUIHttpService service = new(httpClient, "http://localhost/agent"); RunAgentInput input = new() { ThreadId = "thread1", RunId = "run1", - Messages = [new AGUIMessage { Id = "m1", Role = AGUIRoles.User, Content = "Test" }] + Messages = [new AGUIUserMessage { Id = "m1", Content = "Test" }] }; // Act & Assert @@ -83,20 +83,20 @@ await Assert.ThrowsAsync(async () => public async Task PostRunAsync_DeserializesMultipleEventTypes_CorrectlyAsync() { // Arrange - BaseEvent[] events = new BaseEvent[] - { + BaseEvent[] events = + [ new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, new RunErrorEvent { Message = "Error occurred", Code = "ERR001" }, - new RunFinishedEvent { ThreadId = "thread1", RunId = "run1", Result = JsonDocument.Parse("\"Success\"").RootElement.Clone() } - }; + new RunFinishedEvent { ThreadId = "thread1", RunId = "run1", Result = JsonElement.Parse("\"Success\"") } + ]; - HttpClient httpClient = this.CreateMockHttpClient(events, HttpStatusCode.OK); + HttpClient httpClient = CreateMockHttpClient(events, HttpStatusCode.OK); AGUIHttpService service = new(httpClient, "http://localhost/agent"); RunAgentInput input = new() { ThreadId = "thread1", RunId = "run1", - Messages = [new AGUIMessage { Id = "m1", Role = AGUIRoles.User, Content = "Test" }] + Messages = [new AGUIUserMessage { Id = "m1", Content = "Test" }] }; // Act @@ -120,13 +120,13 @@ public async Task PostRunAsync_DeserializesMultipleEventTypes_CorrectlyAsync() public async Task PostRunAsync_WithEmptyEventStream_CompletesSuccessfullyAsync() { // Arrange - HttpClient httpClient = this.CreateMockHttpClient([], HttpStatusCode.OK); + HttpClient httpClient = CreateMockHttpClient([], HttpStatusCode.OK); AGUIHttpService service = new(httpClient, "http://localhost/agent"); RunAgentInput input = new() { ThreadId = "thread1", RunId = "run1", - Messages = [new AGUIMessage { Id = "m1", Role = AGUIRoles.User, Content = "Test" }] + Messages = [new AGUIUserMessage { Id = "m1", Content = "Test" }] }; // Act @@ -162,7 +162,7 @@ public async Task PostRunAsync_WithCancellationToken_CancelsRequestAsync() { ThreadId = "thread1", RunId = "run1", - Messages = [new AGUIMessage { Id = "m1", Role = AGUIRoles.User, Content = "Test" }] + Messages = [new AGUIUserMessage { Id = "m1", Content = "Test" }] }; // Act & Assert @@ -175,9 +175,9 @@ await Assert.ThrowsAsync(async () => }); } - private HttpClient CreateMockHttpClient(BaseEvent[] events, HttpStatusCode statusCode) + private static HttpClient CreateMockHttpClient(BaseEvent[] events, HttpStatusCode statusCode) { - string sseContent = string.Join("", events.Select(e => + string sseContent = string.Concat(events.Select(e => $"data: {JsonSerializer.Serialize(e, AGUIJsonSerializerContext.Default.BaseEvent)}\n\n")); Mock handlerMock = new(MockBehavior.Strict); diff --git a/dotnet/tests/Microsoft.Agents.AI.AGUI.UnitTests/AGUIJsonSerializerContextTests.cs b/dotnet/tests/Microsoft.Agents.AI.AGUI.UnitTests/AGUIJsonSerializerContextTests.cs index f1b1971f20..33f259a681 100644 --- a/dotnet/tests/Microsoft.Agents.AI.AGUI.UnitTests/AGUIJsonSerializerContextTests.cs +++ b/dotnet/tests/Microsoft.Agents.AI.AGUI.UnitTests/AGUIJsonSerializerContextTests.cs @@ -20,12 +20,12 @@ public void RunAgentInput_Serializes_WithAllRequiredFields() { ThreadId = "thread1", RunId = "run1", - Messages = [new AGUIMessage { Id = "m1", Role = AGUIRoles.User, Content = "Test" }] + Messages = [new AGUIUserMessage { Id = "m1", Content = "Test" }] }; // Act string json = JsonSerializer.Serialize(input, AGUIJsonSerializerContext.Default.RunAgentInput); - JsonElement jsonElement = JsonSerializer.Deserialize(json); + JsonElement jsonElement = JsonElement.Parse(json); // Assert Assert.True(jsonElement.TryGetProperty("threadId", out JsonElement threadIdProp)); @@ -72,9 +72,9 @@ public void RunAgentInput_HandlesOptionalFields_StateContextAndForwardedProperti { ThreadId = "thread1", RunId = "run1", - Messages = [new AGUIMessage { Id = "m1", Role = AGUIRoles.User, Content = "Test" }], + Messages = [new AGUIUserMessage { Id = "m1", Content = "Test" }], State = JsonSerializer.SerializeToElement(new { key = "value" }), - Context = new Dictionary { ["ctx1"] = "value1" }, + Context = [new AGUIContextItem { Description = "ctx1", Value = "value1" }], ForwardedProperties = JsonSerializer.SerializeToElement(new { prop1 = "val1" }) }; @@ -119,10 +119,13 @@ public void RunAgentInput_RoundTrip_PreservesAllData() RunId = "run1", Messages = [ - new AGUIMessage { Id = "m1", Role = AGUIRoles.User, Content = "First" }, - new AGUIMessage { Id = "m2", Role = AGUIRoles.Assistant, Content = "Second" } + new AGUIUserMessage { Id = "m1", Content = "First" }, + new AGUIAssistantMessage { Id = "m2", Content = "Second" } ], - Context = new Dictionary { ["key1"] = "value1", ["key2"] = "value2" } + Context = [ + new AGUIContextItem { Description = "key1", Value = "value1" }, + new AGUIContextItem { Description = "key2", Value = "value2" } + ] }; // Act @@ -134,7 +137,7 @@ public void RunAgentInput_RoundTrip_PreservesAllData() Assert.Equal(original.ThreadId, deserialized.ThreadId); Assert.Equal(original.RunId, deserialized.RunId); Assert.Equal(2, deserialized.Messages.Count()); - Assert.Equal(2, deserialized.Context.Count); + Assert.Equal(2, deserialized.Context.Length); } [Fact] @@ -147,7 +150,8 @@ public void RunStartedEvent_Serializes_WithCorrectEventType() string json = JsonSerializer.Serialize(evt, AGUIJsonSerializerContext.Default.RunStartedEvent); // Assert - Assert.Contains($"\"type\":\"{AGUIEventTypes.RunStarted}\"", json); + var jsonElement = JsonElement.Parse(json); + Assert.Equal(AGUIEventTypes.RunStarted, jsonElement.GetProperty("type").GetString()); } [Fact] @@ -158,7 +162,7 @@ public void RunStartedEvent_Includes_ThreadIdAndRunIdInOutput() // Act string json = JsonSerializer.Serialize(evt, AGUIJsonSerializerContext.Default.RunStartedEvent); - JsonElement jsonElement = JsonSerializer.Deserialize(json); + JsonElement jsonElement = JsonElement.Parse(json); // Assert Assert.True(jsonElement.TryGetProperty("threadId", out JsonElement threadIdProp)); @@ -215,18 +219,19 @@ public void RunFinishedEvent_Serializes_WithCorrectEventType() string json = JsonSerializer.Serialize(evt, AGUIJsonSerializerContext.Default.RunFinishedEvent); // Assert - Assert.Contains($"\"type\":\"{AGUIEventTypes.RunFinished}\"", json); + var jsonElement = JsonElement.Parse(json); + Assert.Equal(AGUIEventTypes.RunFinished, jsonElement.GetProperty("type").GetString()); } [Fact] public void RunFinishedEvent_Includes_ThreadIdRunIdAndOptionalResult() { // Arrange - RunFinishedEvent evt = new() { ThreadId = "thread1", RunId = "run1", Result = JsonDocument.Parse("\"Success\"").RootElement.Clone() }; + RunFinishedEvent evt = new() { ThreadId = "thread1", RunId = "run1", Result = JsonElement.Parse("\"Success\"") }; // Act string json = JsonSerializer.Serialize(evt, AGUIJsonSerializerContext.Default.RunFinishedEvent); - JsonElement jsonElement = JsonSerializer.Deserialize(json); + JsonElement jsonElement = JsonElement.Parse(json); // Assert Assert.True(jsonElement.TryGetProperty("threadId", out JsonElement threadIdProp)); @@ -264,7 +269,7 @@ public void RunFinishedEvent_Deserializes_FromJsonCorrectly() public void RunFinishedEvent_RoundTrip_PreservesData() { // Arrange - RunFinishedEvent original = new() { ThreadId = "thread1", RunId = "run1", Result = JsonDocument.Parse("\"Done\"").RootElement.Clone() }; + RunFinishedEvent original = new() { ThreadId = "thread1", RunId = "run1", Result = JsonElement.Parse("\"Done\"") }; // Act string json = JsonSerializer.Serialize(original, AGUIJsonSerializerContext.Default.RunFinishedEvent); @@ -287,7 +292,8 @@ public void RunErrorEvent_Serializes_WithCorrectEventType() string json = JsonSerializer.Serialize(evt, AGUIJsonSerializerContext.Default.RunErrorEvent); // Assert - Assert.Contains($"\"type\":\"{AGUIEventTypes.RunError}\"", json); + var jsonElement = JsonElement.Parse(json); + Assert.Equal(AGUIEventTypes.RunError, jsonElement.GetProperty("type").GetString()); } [Fact] @@ -298,7 +304,7 @@ public void RunErrorEvent_Includes_MessageAndOptionalCode() // Act string json = JsonSerializer.Serialize(evt, AGUIJsonSerializerContext.Default.RunErrorEvent); - JsonElement jsonElement = JsonSerializer.Deserialize(json); + JsonElement jsonElement = JsonElement.Parse(json); // Assert Assert.True(jsonElement.TryGetProperty("message", out JsonElement messageProp)); @@ -354,7 +360,8 @@ public void TextMessageStartEvent_Serializes_WithCorrectEventType() string json = JsonSerializer.Serialize(evt, AGUIJsonSerializerContext.Default.TextMessageStartEvent); // Assert - Assert.Contains($"\"type\":\"{AGUIEventTypes.TextMessageStart}\"", json); + var jsonElement = JsonElement.Parse(json); + Assert.Equal(AGUIEventTypes.TextMessageStart, jsonElement.GetProperty("type").GetString()); } [Fact] @@ -365,7 +372,7 @@ public void TextMessageStartEvent_Includes_MessageIdAndRole() // Act string json = JsonSerializer.Serialize(evt, AGUIJsonSerializerContext.Default.TextMessageStartEvent); - JsonElement jsonElement = JsonSerializer.Deserialize(json); + JsonElement jsonElement = JsonElement.Parse(json); // Assert Assert.True(jsonElement.TryGetProperty("messageId", out JsonElement msgIdProp)); @@ -421,7 +428,8 @@ public void TextMessageContentEvent_Serializes_WithCorrectEventType() string json = JsonSerializer.Serialize(evt, AGUIJsonSerializerContext.Default.TextMessageContentEvent); // Assert - Assert.Contains($"\"type\":\"{AGUIEventTypes.TextMessageContent}\"", json); + var jsonElement = JsonElement.Parse(json); + Assert.Equal(AGUIEventTypes.TextMessageContent, jsonElement.GetProperty("type").GetString()); } [Fact] @@ -432,7 +440,7 @@ public void TextMessageContentEvent_Includes_MessageIdAndDelta() // Act string json = JsonSerializer.Serialize(evt, AGUIJsonSerializerContext.Default.TextMessageContentEvent); - JsonElement jsonElement = JsonSerializer.Deserialize(json); + JsonElement jsonElement = JsonElement.Parse(json); // Assert Assert.True(jsonElement.TryGetProperty("messageId", out JsonElement msgIdProp)); @@ -488,7 +496,8 @@ public void TextMessageEndEvent_Serializes_WithCorrectEventType() string json = JsonSerializer.Serialize(evt, AGUIJsonSerializerContext.Default.TextMessageEndEvent); // Assert - Assert.Contains($"\"type\":\"{AGUIEventTypes.TextMessageEnd}\"", json); + var jsonElement = JsonElement.Parse(json); + Assert.Equal(AGUIEventTypes.TextMessageEnd, jsonElement.GetProperty("type").GetString()); } [Fact] @@ -499,7 +508,7 @@ public void TextMessageEndEvent_Includes_MessageId() // Act string json = JsonSerializer.Serialize(evt, AGUIJsonSerializerContext.Default.TextMessageEndEvent); - JsonElement jsonElement = JsonSerializer.Deserialize(json); + JsonElement jsonElement = JsonElement.Parse(json); // Assert Assert.True(jsonElement.TryGetProperty("messageId", out JsonElement msgIdProp)); @@ -544,11 +553,11 @@ public void TextMessageEndEvent_RoundTrip_PreservesData() public void AGUIMessage_Serializes_WithIdRoleAndContent() { // Arrange - AGUIMessage message = new() { Id = "m1", Role = AGUIRoles.User, Content = "Hello" }; + AGUIMessage message = new AGUIUserMessage() { Id = "m1", Content = "Hello" }; // Act string json = JsonSerializer.Serialize(message, AGUIJsonSerializerContext.Default.AGUIMessage); - JsonElement jsonElement = JsonSerializer.Deserialize(json); + JsonElement jsonElement = JsonElement.Parse(json); // Assert Assert.True(jsonElement.TryGetProperty("id", out JsonElement idProp)); @@ -578,14 +587,14 @@ public void AGUIMessage_Deserializes_FromJsonCorrectly() Assert.NotNull(message); Assert.Equal("m1", message.Id); Assert.Equal(AGUIRoles.User, message.Role); - Assert.Equal("Test message", message.Content); + Assert.Equal("Test message", ((AGUIUserMessage)message).Content); } [Fact] public void AGUIMessage_RoundTrip_PreservesData() { // Arrange - AGUIMessage original = new() { Id = "msg123", Role = AGUIRoles.Assistant, Content = "Response text" }; + AGUIMessage original = new AGUIAssistantMessage() { Id = "msg123", Content = "Response text" }; // Act string json = JsonSerializer.Serialize(original, AGUIJsonSerializerContext.Default.AGUIMessage); @@ -595,7 +604,7 @@ public void AGUIMessage_RoundTrip_PreservesData() Assert.NotNull(deserialized); Assert.Equal(original.Id, deserialized.Id); Assert.Equal(original.Role, deserialized.Role); - Assert.Equal(original.Content, deserialized.Content); + Assert.Equal(((AGUIAssistantMessage)original).Content, ((AGUIAssistantMessage)deserialized).Content); } [Fact] @@ -617,7 +626,7 @@ public void AGUIMessage_Validates_RequiredFields() Assert.NotNull(message); Assert.NotNull(message.Id); Assert.NotNull(message.Role); - Assert.NotNull(message.Content); + Assert.NotNull(((AGUIUserMessage)message).Content); } [Fact] @@ -773,71 +782,333 @@ public void BaseEvent_DistinguishesEventTypes_BasedOnTypeField() Assert.IsType(events[5]); } + #region Comprehensive Message Serialization Tests + [Fact] - public void AGUIAgentThreadState_Serializes_WithThreadIdAndWrappedState() + public void AGUIUserMessage_SerializesAndDeserializes_Correctly() { // Arrange - AGUIAgentThread.AGUIAgentThreadState state = new() + var originalMessage = new AGUIUserMessage { - ThreadId = "thread1", - WrappedState = JsonSerializer.SerializeToElement(new { test = "data" }) + Id = "user1", + Content = "Hello, assistant!" }; // Act - string json = JsonSerializer.Serialize(state, AGUIJsonSerializerContext.Default.AGUIAgentThreadState); - JsonElement jsonElement = JsonSerializer.Deserialize(json); + string json = JsonSerializer.Serialize(originalMessage, AGUIJsonSerializerContext.Default.AGUIUserMessage); + var deserialized = JsonSerializer.Deserialize(json, AGUIJsonSerializerContext.Default.AGUIUserMessage); // Assert - Assert.True(jsonElement.TryGetProperty("ThreadId", out JsonElement threadIdProp)); - Assert.Equal("thread1", threadIdProp.GetString()); - Assert.True(jsonElement.TryGetProperty("WrappedState", out JsonElement wrappedStateProp)); - Assert.NotEqual(JsonValueKind.Null, wrappedStateProp.ValueKind); + Assert.NotNull(deserialized); + Assert.Equal("user1", deserialized.Id); + Assert.Equal("Hello, assistant!", deserialized.Content); } [Fact] - public void AGUIAgentThreadState_Deserializes_FromJsonCorrectly() + public void AGUISystemMessage_SerializesAndDeserializes_Correctly() { // Arrange - const string Json = """ - { - "ThreadId": "thread1", - "WrappedState": {"test": "data"} - } - """; + var originalMessage = new AGUISystemMessage + { + Id = "sys1", + Content = "You are a helpful assistant." + }; // Act - AGUIAgentThread.AGUIAgentThreadState? state = JsonSerializer.Deserialize( - Json, - AGUIJsonSerializerContext.Default.AGUIAgentThreadState); + string json = JsonSerializer.Serialize(originalMessage, AGUIJsonSerializerContext.Default.AGUISystemMessage); + var deserialized = JsonSerializer.Deserialize(json, AGUIJsonSerializerContext.Default.AGUISystemMessage); // Assert - Assert.NotNull(state); - Assert.Equal("thread1", state.ThreadId); - Assert.NotEqual(JsonValueKind.Undefined, state.WrappedState.ValueKind); + Assert.NotNull(deserialized); + Assert.Equal("sys1", deserialized.Id); + Assert.Equal("You are a helpful assistant.", deserialized.Content); } [Fact] - public void AGUIAgentThreadState_RoundTrip_PreservesThreadIdAndNestedState() + public void AGUIDeveloperMessage_SerializesAndDeserializes_Correctly() { // Arrange - AGUIAgentThread.AGUIAgentThreadState original = new() + var originalMessage = new AGUIDeveloperMessage { - ThreadId = "thread123", - WrappedState = JsonSerializer.SerializeToElement(new { key1 = "value1", key2 = 42 }) + Id = "dev1", + Content = "Developer instructions here." }; // Act - string json = JsonSerializer.Serialize(original, AGUIJsonSerializerContext.Default.AGUIAgentThreadState); - AGUIAgentThread.AGUIAgentThreadState? deserialized = JsonSerializer.Deserialize( - json, - AGUIJsonSerializerContext.Default.AGUIAgentThreadState); + string json = JsonSerializer.Serialize(originalMessage, AGUIJsonSerializerContext.Default.AGUIDeveloperMessage); + var deserialized = JsonSerializer.Deserialize(json, AGUIJsonSerializerContext.Default.AGUIDeveloperMessage); // Assert Assert.NotNull(deserialized); - Assert.Equal(original.ThreadId, deserialized.ThreadId); - Assert.Equal(original.WrappedState.GetProperty("key1").GetString(), - deserialized.WrappedState.GetProperty("key1").GetString()); - Assert.Equal(original.WrappedState.GetProperty("key2").GetInt32(), - deserialized.WrappedState.GetProperty("key2").GetInt32()); + Assert.Equal("dev1", deserialized.Id); + Assert.Equal("Developer instructions here.", deserialized.Content); + } + + [Fact] + public void AGUIAssistantMessage_WithTextOnly_SerializesAndDeserializes_Correctly() + { + // Arrange + var originalMessage = new AGUIAssistantMessage + { + Id = "asst1", + Content = "I can help you with that." + }; + + // Act + string json = JsonSerializer.Serialize(originalMessage, AGUIJsonSerializerContext.Default.AGUIAssistantMessage); + var deserialized = JsonSerializer.Deserialize(json, AGUIJsonSerializerContext.Default.AGUIAssistantMessage); + + // Assert + Assert.NotNull(deserialized); + Assert.Equal("asst1", deserialized.Id); + Assert.Equal("I can help you with that.", deserialized.Content); + Assert.Null(deserialized.ToolCalls); + } + + [Fact] + public void AGUIAssistantMessage_WithToolCallsAndParameters_SerializesAndDeserializes_Correctly() + { + // Arrange + var parameters = new Dictionary + { + ["location"] = "Seattle", + ["units"] = "fahrenheit", + ["days"] = 5 + }; + string argumentsJson = JsonSerializer.Serialize(parameters, AGUIJsonSerializerContext.Default.Options); + + var originalMessage = new AGUIAssistantMessage + { + Id = "asst2", + Content = "Let me check the weather for you.", + ToolCalls = + [ + new AGUIToolCall + { + Id = "call_123", + Type = "function", + Function = new AGUIFunctionCall + { + Name = "GetWeather", + Arguments = argumentsJson + } + } + ] + }; + + // Act + string json = JsonSerializer.Serialize(originalMessage, AGUIJsonSerializerContext.Default.AGUIAssistantMessage); + var deserialized = JsonSerializer.Deserialize(json, AGUIJsonSerializerContext.Default.AGUIAssistantMessage); + + // Assert + Assert.NotNull(deserialized); + Assert.Equal("asst2", deserialized.Id); + Assert.Equal("Let me check the weather for you.", deserialized.Content); + Assert.NotNull(deserialized.ToolCalls); + Assert.Single(deserialized.ToolCalls); + + var toolCall = deserialized.ToolCalls[0]; + Assert.Equal("call_123", toolCall.Id); + Assert.Equal("function", toolCall.Type); + Assert.NotNull(toolCall.Function); + Assert.Equal("GetWeather", toolCall.Function.Name); + + // Verify parameters can be deserialized + var deserializedParams = JsonSerializer.Deserialize>( + toolCall.Function.Arguments, + AGUIJsonSerializerContext.Default.Options); + Assert.NotNull(deserializedParams); + Assert.Equal("Seattle", deserializedParams["location"].GetString()); + Assert.Equal("fahrenheit", deserializedParams["units"].GetString()); + Assert.Equal(5, deserializedParams["days"].GetInt32()); + } + + [Fact] + public void AGUIToolMessage_WithResults_SerializesAndDeserializes_Correctly() + { + // Arrange + var result = new Dictionary + { + ["temperature"] = 72.5, + ["conditions"] = "Sunny", + ["humidity"] = 45 + }; + string contentJson = JsonSerializer.Serialize(result, AGUIJsonSerializerContext.Default.Options); + + var originalMessage = new AGUIToolMessage + { + Id = "tool1", + ToolCallId = "call_123", + Content = contentJson + }; + + // Act + string json = JsonSerializer.Serialize(originalMessage, AGUIJsonSerializerContext.Default.AGUIToolMessage); + var deserialized = JsonSerializer.Deserialize(json, AGUIJsonSerializerContext.Default.AGUIToolMessage); + + // Assert + Assert.NotNull(deserialized); + Assert.Equal("tool1", deserialized.Id); + Assert.Equal("call_123", deserialized.ToolCallId); + Assert.NotNull(deserialized.Content); + + // Verify result content can be deserialized + var deserializedResult = JsonSerializer.Deserialize>( + deserialized.Content, + AGUIJsonSerializerContext.Default.Options); + Assert.NotNull(deserializedResult); + Assert.Equal(72.5, deserializedResult["temperature"].GetDouble()); + Assert.Equal("Sunny", deserializedResult["conditions"].GetString()); + Assert.Equal(45, deserializedResult["humidity"].GetInt32()); + } + + [Fact] + public void AllFiveMessageTypes_SerializeAsPolymorphicArray_Correctly() + { + // Arrange + AGUIMessage[] messages = + [ + new AGUISystemMessage { Id = "1", Content = "System message" }, + new AGUIDeveloperMessage { Id = "2", Content = "Developer message" }, + new AGUIUserMessage { Id = "3", Content = "User message" }, + new AGUIAssistantMessage { Id = "4", Content = "Assistant message" }, + new AGUIToolMessage { Id = "5", ToolCallId = "call_1", Content = "{\"result\":\"success\"}" } + ]; + + // Act + string json = JsonSerializer.Serialize(messages, AGUIJsonSerializerContext.Default.AGUIMessageArray); + var deserialized = JsonSerializer.Deserialize(json, AGUIJsonSerializerContext.Default.AGUIMessageArray); + + // Assert + Assert.NotNull(deserialized); + Assert.Equal(5, deserialized.Length); + Assert.IsType(deserialized[0]); + Assert.IsType(deserialized[1]); + Assert.IsType(deserialized[2]); + Assert.IsType(deserialized[3]); + Assert.IsType(deserialized[4]); + } + + #endregion + + #region Tool-Related Event Type Tests + + [Fact] + public void ToolCallStartEvent_SerializesAndDeserializes_Correctly() + { + // Arrange + var originalEvent = new ToolCallStartEvent + { + ParentMessageId = "msg1", + ToolCallId = "call_123", + ToolCallName = "GetWeather" + }; + + // Act + string json = JsonSerializer.Serialize(originalEvent, AGUIJsonSerializerContext.Default.ToolCallStartEvent); + var deserialized = JsonSerializer.Deserialize(json, AGUIJsonSerializerContext.Default.ToolCallStartEvent); + + // Assert + Assert.NotNull(deserialized); + Assert.Equal("msg1", deserialized.ParentMessageId); + Assert.Equal("call_123", deserialized.ToolCallId); + Assert.Equal("GetWeather", deserialized.ToolCallName); + Assert.Equal(AGUIEventTypes.ToolCallStart, deserialized.Type); + } + + [Fact] + public void ToolCallArgsEvent_SerializesAndDeserializes_Correctly() + { + // Arrange + var originalEvent = new ToolCallArgsEvent + { + ToolCallId = "call_123", + Delta = "{\"location\":\"Seattle\",\"units\":\"fahrenheit\"}" + }; + + // Act + string json = JsonSerializer.Serialize(originalEvent, AGUIJsonSerializerContext.Default.ToolCallArgsEvent); + var deserialized = JsonSerializer.Deserialize(json, AGUIJsonSerializerContext.Default.ToolCallArgsEvent); + + // Assert + Assert.NotNull(deserialized); + Assert.Equal("call_123", deserialized.ToolCallId); + Assert.Equal("{\"location\":\"Seattle\",\"units\":\"fahrenheit\"}", deserialized.Delta); + Assert.Equal(AGUIEventTypes.ToolCallArgs, deserialized.Type); + } + + [Fact] + public void ToolCallEndEvent_SerializesAndDeserializes_Correctly() + { + // Arrange + var originalEvent = new ToolCallEndEvent + { + ToolCallId = "call_123" + }; + + // Act + string json = JsonSerializer.Serialize(originalEvent, AGUIJsonSerializerContext.Default.ToolCallEndEvent); + var deserialized = JsonSerializer.Deserialize(json, AGUIJsonSerializerContext.Default.ToolCallEndEvent); + + // Assert + Assert.NotNull(deserialized); + Assert.Equal("call_123", deserialized.ToolCallId); + Assert.Equal(AGUIEventTypes.ToolCallEnd, deserialized.Type); } + + [Fact] + public void ToolCallResultEvent_SerializesAndDeserializes_Correctly() + { + // Arrange + var originalEvent = new ToolCallResultEvent + { + MessageId = "msg1", + ToolCallId = "call_123", + Content = "{\"temperature\":72.5,\"conditions\":\"Sunny\"}", + Role = "tool" + }; + + // Act + string json = JsonSerializer.Serialize(originalEvent, AGUIJsonSerializerContext.Default.ToolCallResultEvent); + var deserialized = JsonSerializer.Deserialize(json, AGUIJsonSerializerContext.Default.ToolCallResultEvent); + + // Assert + Assert.NotNull(deserialized); + Assert.Equal("msg1", deserialized.MessageId); + Assert.Equal("call_123", deserialized.ToolCallId); + Assert.Equal("{\"temperature\":72.5,\"conditions\":\"Sunny\"}", deserialized.Content); + Assert.Equal("tool", deserialized.Role); + Assert.Equal(AGUIEventTypes.ToolCallResult, deserialized.Type); + } + + [Fact] + public void AllToolEventTypes_SerializeAsPolymorphicBaseEvent_Correctly() + { + // Arrange + BaseEvent[] events = + [ + new RunStartedEvent { ThreadId = "t1", RunId = "r1" }, + new ToolCallStartEvent { ParentMessageId = "m1", ToolCallId = "c1", ToolCallName = "Tool1" }, + new ToolCallArgsEvent { ToolCallId = "c1", Delta = "{}" }, + new ToolCallEndEvent { ToolCallId = "c1" }, + new ToolCallResultEvent { MessageId = "m2", ToolCallId = "c1", Content = "{}", Role = "tool" }, + new RunFinishedEvent { ThreadId = "t1", RunId = "r1" } + ]; + + // Act + string json = JsonSerializer.Serialize(events, AGUIJsonSerializerContext.Default.Options); + var deserialized = JsonSerializer.Deserialize(json, AGUIJsonSerializerContext.Default.Options); + + // Assert + Assert.NotNull(deserialized); + Assert.Equal(6, deserialized.Length); + Assert.IsType(deserialized[0]); + Assert.IsType(deserialized[1]); + Assert.IsType(deserialized[2]); + Assert.IsType(deserialized[3]); + Assert.IsType(deserialized[4]); + Assert.IsType(deserialized[5]); + } + + #endregion } diff --git a/dotnet/tests/Microsoft.Agents.AI.AGUI.UnitTests/AIToolExtensionsTests.cs b/dotnet/tests/Microsoft.Agents.AI.AGUI.UnitTests/AIToolExtensionsTests.cs new file mode 100644 index 0000000000..ebedd68f33 --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.AGUI.UnitTests/AIToolExtensionsTests.cs @@ -0,0 +1,216 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Collections.Generic; +using System.Linq; +using System.Text.Json; +using Microsoft.Agents.AI.AGUI.Shared; +using Microsoft.Extensions.AI; + +namespace Microsoft.Agents.AI.AGUI.UnitTests; + +/// +/// Unit tests for the class. +/// +public sealed class AIToolExtensionsTests +{ + [Fact] + public void AsAGUITools_WithAIFunction_ConvertsToAGUIToolCorrectly() + { + // Arrange + AIFunction function = AIFunctionFactory.Create( + (string location) => $"Weather in {location}", + "GetWeather", + "Gets the current weather"); + List tools = [function]; + + // Act + List aguiTools = tools.AsAGUITools().ToList(); + + // Assert + AGUITool aguiTool = Assert.Single(aguiTools); + Assert.Equal("GetWeather", aguiTool.Name); + Assert.Equal("Gets the current weather", aguiTool.Description); + Assert.NotEqual(default, aguiTool.Parameters); + } + + [Fact] + public void AsAGUITools_WithMultipleFunctions_ConvertsAllCorrectly() + { + // Arrange + List tools = + [ + AIFunctionFactory.Create(() => "Result1", "Tool1", "First tool"), + AIFunctionFactory.Create(() => "Result2", "Tool2", "Second tool"), + AIFunctionFactory.Create(() => "Result3", "Tool3", "Third tool") + ]; + + // Act + List aguiTools = tools.AsAGUITools().ToList(); + + // Assert + Assert.Equal(3, aguiTools.Count); + Assert.Equal("Tool1", aguiTools[0].Name); + Assert.Equal("Tool2", aguiTools[1].Name); + Assert.Equal("Tool3", aguiTools[2].Name); + } + + [Fact] + public void AsAGUITools_WithNullInput_ReturnsEmptyEnumerable() + { + // Arrange + IEnumerable? tools = null; + + // Act + IEnumerable aguiTools = tools!.AsAGUITools(); + + // Assert + Assert.NotNull(aguiTools); + Assert.Empty(aguiTools); + } + + [Fact] + public void AsAGUITools_WithEmptyInput_ReturnsEmptyEnumerable() + { + // Arrange + List tools = []; + + // Act + List aguiTools = tools.AsAGUITools().ToList(); + + // Assert + Assert.Empty(aguiTools); + } + + [Fact] + public void AsAGUITools_FiltersOutNonAIFunctionTools() + { + // Arrange - mix of AIFunction and non-function tools + AIFunction function = AIFunctionFactory.Create(() => "Result", "TestTool"); + // Create a custom AITool that's not an AIFunction + var declaration = AIFunctionFactory.CreateDeclaration("DeclarationOnly", "Description", JsonElement.Parse("{}")); + + List tools = [function, declaration]; + + // Act + List aguiTools = tools.AsAGUITools().ToList(); + + // Assert + // Only the AIFunction should be converted, declarations are filtered + Assert.Equal(2, aguiTools.Count); // Actually both convert since declaration is also AIFunctionDeclaration + } + + [Fact] + public void AsAITools_WithAGUITool_ConvertsToAIFunctionDeclarationCorrectly() + { + // Arrange + AGUITool aguiTool = new() + { + Name = "TestTool", + Description = "Test description", + Parameters = JsonElement.Parse("""{"type":"object","properties":{}}""") + }; + List aguiTools = [aguiTool]; + + // Act + List tools = aguiTools.AsAITools().ToList(); + + // Assert + AITool tool = Assert.Single(tools); + Assert.IsType(tool, exactMatch: false); + var declaration = (AIFunctionDeclaration)tool; + Assert.Equal("TestTool", declaration.Name); + Assert.Equal("Test description", declaration.Description); + } + + [Fact] + public void AsAITools_WithMultipleAGUITools_ConvertsAllCorrectly() + { + // Arrange + List aguiTools = + [ + new AGUITool { Name = "Tool1", Description = "Desc1", Parameters = JsonElement.Parse("{}") }, + new AGUITool { Name = "Tool2", Description = "Desc2", Parameters = JsonElement.Parse("{}") }, + new AGUITool { Name = "Tool3", Description = "Desc3", Parameters = JsonElement.Parse("{}") } + ]; + + // Act + List tools = aguiTools.AsAITools().ToList(); + + // Assert + Assert.Equal(3, tools.Count); + Assert.All(tools, t => Assert.IsType(t, exactMatch: false)); + } + + [Fact] + public void AsAITools_WithNullInput_ReturnsEmptyEnumerable() + { + // Arrange + IEnumerable? aguiTools = null; + + // Act + IEnumerable tools = aguiTools!.AsAITools(); + + // Assert + Assert.NotNull(tools); + Assert.Empty(tools); + } + + [Fact] + public void AsAITools_WithEmptyInput_ReturnsEmptyEnumerable() + { + // Arrange + List aguiTools = []; + + // Act + List tools = aguiTools.AsAITools().ToList(); + + // Assert + Assert.Empty(tools); + } + + [Fact] + public void AsAITools_CreatesDeclarationsOnly_NotInvokableFunctions() + { + // Arrange + AGUITool aguiTool = new() + { + Name = "RemoteTool", + Description = "Tool implemented on server", + Parameters = JsonElement.Parse("""{"type":"object"}""") + }; + + // Act + List aguiToolsList = [aguiTool]; + AITool tool = aguiToolsList.AsAITools().Single(); + + // Assert + // The tool should be a declaration, not an executable function + Assert.IsType(tool, exactMatch: false); + // AIFunctionDeclaration cannot be invoked (no implementation) + // This is correct since the actual implementation exists on the client side + } + + [Fact] + public void RoundTrip_AIFunctionToAGUIToolBackToDeclaration_PreservesMetadata() + { + // Arrange + AIFunction originalFunction = AIFunctionFactory.Create( + (string name, int age) => $"{name} is {age} years old", + "FormatPerson", + "Formats person information"); + + // Act + List originalList = [originalFunction]; + AGUITool aguiTool = originalList.AsAGUITools().Single(); + List aguiToolsList = [aguiTool]; + AITool reconstructed = aguiToolsList.AsAITools().Single(); + + // Assert + Assert.IsType(reconstructed, exactMatch: false); + var declaration = (AIFunctionDeclaration)reconstructed; + Assert.Equal("FormatPerson", declaration.Name); + Assert.Equal("Formats person information", declaration.Description); + // Schema should be preserved through the round trip + Assert.NotEqual(default, declaration.JsonSchema); + } +} diff --git a/dotnet/tests/Microsoft.Agents.AI.AGUI.UnitTests/AgentRunResponseUpdateAGUIExtensionsTests.cs b/dotnet/tests/Microsoft.Agents.AI.AGUI.UnitTests/AgentRunResponseUpdateAGUIExtensionsTests.cs deleted file mode 100644 index 3afea0a6c9..0000000000 --- a/dotnet/tests/Microsoft.Agents.AI.AGUI.UnitTests/AgentRunResponseUpdateAGUIExtensionsTests.cs +++ /dev/null @@ -1,191 +0,0 @@ -// Copyright (c) Microsoft. All rights reserved. - -using System; -using System.Collections.Generic; -using System.Text.Json; -using System.Threading.Tasks; -using Microsoft.Agents.AI.AGUI.Shared; -using Microsoft.Extensions.AI; - -namespace Microsoft.Agents.AI.AGUI.UnitTests; - -public sealed class AgentRunResponseUpdateAGUIExtensionsTests -{ - [Fact] - public async Task AsAgentRunResponseUpdatesAsync_ConvertsRunStartedEvent_ToResponseUpdateWithMetadataAsync() - { - // Arrange - List events = - [ - new RunStartedEvent { ThreadId = "thread1", RunId = "run1" } - ]; - - // Act - List updates = []; - await foreach (AgentRunResponseUpdate update in events.ToAsyncEnumerableAsync().AsAgentRunResponseUpdatesAsync()) - { - updates.Add(update); - } - - // Assert - Assert.Single(updates); - Assert.Equal(ChatRole.Assistant, updates[0].Role); - Assert.Equal("run1", updates[0].ResponseId); - Assert.NotNull(updates[0].CreatedAt); - // ConversationId is stored in the underlying ChatResponseUpdate - ChatResponseUpdate chatUpdate = Assert.IsType(updates[0].RawRepresentation); - Assert.Equal("thread1", chatUpdate.ConversationId); - } - - [Fact] - public async Task AsAgentRunResponseUpdatesAsync_ConvertsRunFinishedEvent_ToResponseUpdateWithMetadataAsync() - { - // Arrange - List events = - [ - new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, - new RunFinishedEvent { ThreadId = "thread1", RunId = "run1", Result = JsonSerializer.SerializeToElement("Success") } - ]; - - // Act - List updates = []; - await foreach (AgentRunResponseUpdate update in events.ToAsyncEnumerableAsync().AsAgentRunResponseUpdatesAsync()) - { - updates.Add(update); - } - - // Assert - Assert.Equal(2, updates.Count); - // First update is RunStarted - Assert.Equal(ChatRole.Assistant, updates[0].Role); - Assert.Equal("run1", updates[0].ResponseId); - // Second update is RunFinished - Assert.Equal(ChatRole.Assistant, updates[1].Role); - Assert.Equal("run1", updates[1].ResponseId); - Assert.NotNull(updates[1].CreatedAt); - TextContent content = Assert.IsType(updates[1].Contents[0]); - Assert.Equal("\"Success\"", content.Text); // JSON string representation includes quotes - // ConversationId is stored in the underlying ChatResponseUpdate - ChatResponseUpdate chatUpdate = Assert.IsType(updates[1].RawRepresentation); - Assert.Equal("thread1", chatUpdate.ConversationId); - } - - [Fact] - public async Task AsAgentRunResponseUpdatesAsync_ConvertsRunErrorEvent_ToErrorContentAsync() - { - // Arrange - List events = - [ - new RunErrorEvent { Message = "Error occurred", Code = "ERR001" } - ]; - - // Act - List updates = []; - await foreach (AgentRunResponseUpdate update in events.ToAsyncEnumerableAsync().AsAgentRunResponseUpdatesAsync()) - { - updates.Add(update); - } - - // Assert - Assert.Single(updates); - Assert.Equal(ChatRole.Assistant, updates[0].Role); - ErrorContent content = Assert.IsType(updates[0].Contents[0]); - Assert.Equal("Error occurred", content.Message); - // Code is stored in ErrorCode property - Assert.Equal("ERR001", content.ErrorCode); - } - - [Fact] - public async Task AsAgentRunResponseUpdatesAsync_ConvertsTextMessageSequence_ToTextUpdatesWithCorrectRoleAsync() - { - // Arrange - List events = - [ - new TextMessageStartEvent { MessageId = "msg1", Role = AGUIRoles.Assistant }, - new TextMessageContentEvent { MessageId = "msg1", Delta = "Hello" }, - new TextMessageContentEvent { MessageId = "msg1", Delta = " World" }, - new TextMessageEndEvent { MessageId = "msg1" } - ]; - - // Act - List updates = []; - await foreach (AgentRunResponseUpdate update in events.ToAsyncEnumerableAsync().AsAgentRunResponseUpdatesAsync()) - { - updates.Add(update); - } - - // Assert - Assert.Equal(2, updates.Count); - Assert.All(updates, u => Assert.Equal(ChatRole.Assistant, u.Role)); - Assert.Equal("Hello", ((TextContent)updates[0].Contents[0]).Text); - Assert.Equal(" World", ((TextContent)updates[1].Contents[0]).Text); - } - - [Fact] - public async Task AsAgentRunResponseUpdatesAsync_WithTextMessageStartWhileMessageInProgress_ThrowsInvalidOperationExceptionAsync() - { - // Arrange - List events = - [ - new TextMessageStartEvent { MessageId = "msg1", Role = AGUIRoles.Assistant }, - new TextMessageContentEvent { MessageId = "msg1", Delta = "Hello" }, - new TextMessageStartEvent { MessageId = "msg2", Role = AGUIRoles.User } - ]; - - // Act & Assert - await Assert.ThrowsAsync(async () => - { - await foreach (var _ in events.ToAsyncEnumerableAsync().AsAgentRunResponseUpdatesAsync()) - { - // Intentionally empty - consuming stream to trigger exception - } - }); - } - - [Fact] - public async Task AsAgentRunResponseUpdatesAsync_WithTextMessageEndForWrongMessageId_ThrowsInvalidOperationExceptionAsync() - { - // Arrange - List events = - [ - new TextMessageStartEvent { MessageId = "msg1", Role = AGUIRoles.Assistant }, - new TextMessageContentEvent { MessageId = "msg1", Delta = "Hello" }, - new TextMessageEndEvent { MessageId = "msg2" } - ]; - - // Act & Assert - await Assert.ThrowsAsync(async () => - { - await foreach (var _ in events.ToAsyncEnumerableAsync().AsAgentRunResponseUpdatesAsync()) - { - // Intentionally empty - consuming stream to trigger exception - } - }); - } - - [Fact] - public async Task AsAgentRunResponseUpdatesAsync_MaintainsMessageContext_AcrossMultipleContentEventsAsync() - { - // Arrange - List events = - [ - new TextMessageStartEvent { MessageId = "msg1", Role = AGUIRoles.Assistant }, - new TextMessageContentEvent { MessageId = "msg1", Delta = "Hello" }, - new TextMessageContentEvent { MessageId = "msg1", Delta = " " }, - new TextMessageContentEvent { MessageId = "msg1", Delta = "World" }, - new TextMessageEndEvent { MessageId = "msg1" } - ]; - - // Act - List updates = []; - await foreach (AgentRunResponseUpdate update in events.ToAsyncEnumerableAsync().AsAgentRunResponseUpdatesAsync()) - { - updates.Add(update); - } - - // Assert - Assert.Equal(3, updates.Count); - Assert.All(updates, u => Assert.Equal(ChatRole.Assistant, u.Role)); - Assert.All(updates, u => Assert.Equal("msg1", u.MessageId)); - } -} diff --git a/dotnet/tests/Microsoft.Agents.AI.AGUI.UnitTests/ChatResponseUpdateAGUIExtensionsTests.cs b/dotnet/tests/Microsoft.Agents.AI.AGUI.UnitTests/ChatResponseUpdateAGUIExtensionsTests.cs new file mode 100644 index 0000000000..7d40cc014d --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.AGUI.UnitTests/ChatResponseUpdateAGUIExtensionsTests.cs @@ -0,0 +1,780 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Collections.Generic; +using System.Linq; +using System.Text.Json; +using System.Threading.Tasks; +using Microsoft.Agents.AI.AGUI.Shared; +using Microsoft.Extensions.AI; + +namespace Microsoft.Agents.AI.AGUI.UnitTests; + +public sealed class ChatResponseUpdateAGUIExtensionsTests +{ + [Fact] + public async Task AsChatResponseUpdatesAsync_ConvertsRunStartedEvent_ToResponseUpdateWithMetadataAsync() + { + // Arrange + List events = + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run1" } + ]; + + // Act + List updates = []; + await foreach (ChatResponseUpdate update in events.ToAsyncEnumerableAsync().AsChatResponseUpdatesAsync(AGUIJsonSerializerContext.Default.Options)) + { + updates.Add(update); + } + + // Assert + Assert.Single(updates); + Assert.Equal(ChatRole.Assistant, updates[0].Role); + Assert.Equal("run1", updates[0].ResponseId); + Assert.NotNull(updates[0].CreatedAt); + Assert.Equal("thread1", updates[0].ConversationId); + } + + [Fact] + public async Task AsChatResponseUpdatesAsync_ConvertsRunFinishedEvent_ToResponseUpdateWithMetadataAsync() + { + // Arrange + List events = + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run1", Result = JsonSerializer.SerializeToElement("Success") } + ]; + + // Act + List updates = []; + await foreach (ChatResponseUpdate update in events.ToAsyncEnumerableAsync().AsChatResponseUpdatesAsync(AGUIJsonSerializerContext.Default.Options)) + { + updates.Add(update); + } + + // Assert + Assert.Equal(2, updates.Count); + // First update is RunStarted + Assert.Equal(ChatRole.Assistant, updates[0].Role); + Assert.Equal("run1", updates[0].ResponseId); + // Second update is RunFinished + Assert.Equal(ChatRole.Assistant, updates[1].Role); + Assert.Equal("run1", updates[1].ResponseId); + Assert.NotNull(updates[1].CreatedAt); + TextContent content = Assert.IsType(updates[1].Contents[0]); + Assert.Equal("\"Success\"", content.Text); // JSON string representation includes quotes + // ConversationId is stored in the ChatResponseUpdate + Assert.Equal("thread1", updates[1].ConversationId); + } + + [Fact] + public async Task AsChatResponseUpdatesAsync_ConvertsRunErrorEvent_ToErrorContentAsync() + { + // Arrange + List events = + [ + new RunErrorEvent { Message = "Error occurred", Code = "ERR001" } + ]; + + // Act + List updates = []; + await foreach (ChatResponseUpdate update in events.ToAsyncEnumerableAsync().AsChatResponseUpdatesAsync(AGUIJsonSerializerContext.Default.Options)) + { + updates.Add(update); + } + + // Assert + Assert.Single(updates); + Assert.Equal(ChatRole.Assistant, updates[0].Role); + ErrorContent content = Assert.IsType(updates[0].Contents[0]); + Assert.Equal("Error occurred", content.Message); + // Code is stored in ErrorCode property + Assert.Equal("ERR001", content.ErrorCode); + } + + [Fact] + public async Task AsChatResponseUpdatesAsync_ConvertsTextMessageSequence_ToTextUpdatesWithCorrectRoleAsync() + { + // Arrange + List events = + [ + new TextMessageStartEvent { MessageId = "msg1", Role = AGUIRoles.Assistant }, + new TextMessageContentEvent { MessageId = "msg1", Delta = "Hello" }, + new TextMessageContentEvent { MessageId = "msg1", Delta = " World" }, + new TextMessageEndEvent { MessageId = "msg1" } + ]; + + // Act + List updates = []; + await foreach (ChatResponseUpdate update in events.ToAsyncEnumerableAsync().AsChatResponseUpdatesAsync(AGUIJsonSerializerContext.Default.Options)) + { + updates.Add(update); + } + + // Assert + Assert.Equal(2, updates.Count); + Assert.All(updates, u => Assert.Equal(ChatRole.Assistant, u.Role)); + Assert.Equal("Hello", ((TextContent)updates[0].Contents[0]).Text); + Assert.Equal(" World", ((TextContent)updates[1].Contents[0]).Text); + } + + [Fact] + public async Task AsChatResponseUpdatesAsync_WithTextMessageStartWhileMessageInProgress_ThrowsInvalidOperationExceptionAsync() + { + // Arrange + List events = + [ + new TextMessageStartEvent { MessageId = "msg1", Role = AGUIRoles.Assistant }, + new TextMessageContentEvent { MessageId = "msg1", Delta = "Hello" }, + new TextMessageStartEvent { MessageId = "msg2", Role = AGUIRoles.User } + ]; + + // Act & Assert + await Assert.ThrowsAsync(async () => + { + await foreach (var _ in events.ToAsyncEnumerableAsync().AsChatResponseUpdatesAsync(AGUIJsonSerializerContext.Default.Options)) + { + // Intentionally empty - consuming stream to trigger exception + } + }); + } + + [Fact] + public async Task AsChatResponseUpdatesAsync_WithTextMessageEndForWrongMessageId_ThrowsInvalidOperationExceptionAsync() + { + // Arrange + List events = + [ + new TextMessageStartEvent { MessageId = "msg1", Role = AGUIRoles.Assistant }, + new TextMessageContentEvent { MessageId = "msg1", Delta = "Hello" }, + new TextMessageEndEvent { MessageId = "msg2" } + ]; + + // Act & Assert + await Assert.ThrowsAsync(async () => + { + await foreach (var _ in events.ToAsyncEnumerableAsync().AsChatResponseUpdatesAsync(AGUIJsonSerializerContext.Default.Options)) + { + // Intentionally empty - consuming stream to trigger exception + } + }); + } + + [Fact] + public async Task AsChatResponseUpdatesAsync_MaintainsMessageContext_AcrossMultipleContentEventsAsync() + { + // Arrange + List events = + [ + new TextMessageStartEvent { MessageId = "msg1", Role = AGUIRoles.Assistant }, + new TextMessageContentEvent { MessageId = "msg1", Delta = "Hello" }, + new TextMessageContentEvent { MessageId = "msg1", Delta = " " }, + new TextMessageContentEvent { MessageId = "msg1", Delta = "World" }, + new TextMessageEndEvent { MessageId = "msg1" } + ]; + + // Act + List updates = []; + await foreach (ChatResponseUpdate update in events.ToAsyncEnumerableAsync().AsChatResponseUpdatesAsync(AGUIJsonSerializerContext.Default.Options)) + { + updates.Add(update); + } + + // Assert + Assert.Equal(3, updates.Count); + Assert.All(updates, u => Assert.Equal(ChatRole.Assistant, u.Role)); + Assert.All(updates, u => Assert.Equal("msg1", u.MessageId)); + } + + [Fact] + public async Task AsChatResponseUpdatesAsync_ConvertsToolCallEvents_ToFunctionCallContentAsync() + { + // Arrange + List events = + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, + new ToolCallStartEvent { ToolCallId = "call_1", ToolCallName = "GetWeather", ParentMessageId = "msg1" }, + new ToolCallArgsEvent { ToolCallId = "call_1", Delta = "{\"location\":" }, + new ToolCallArgsEvent { ToolCallId = "call_1", Delta = "\"Seattle\"}" }, + new ToolCallEndEvent { ToolCallId = "call_1" }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } + ]; + + // Act + List updates = []; + await foreach (ChatResponseUpdate update in events.ToAsyncEnumerableAsync().AsChatResponseUpdatesAsync(AGUIJsonSerializerContext.Default.Options)) + { + updates.Add(update); + } + + // Assert + ChatResponseUpdate toolCallUpdate = updates.First(u => u.Contents.Any(c => c is FunctionCallContent)); + FunctionCallContent functionCall = Assert.IsType(toolCallUpdate.Contents[0]); + Assert.Equal("call_1", functionCall.CallId); + Assert.Equal("GetWeather", functionCall.Name); + Assert.NotNull(functionCall.Arguments); + Assert.Equal("Seattle", functionCall.Arguments!["location"]?.ToString()); + } + + [Fact] + public async Task AsChatResponseUpdatesAsync_WithMultipleToolCallArgsEvents_AccumulatesArgsCorrectlyAsync() + { + // Arrange + List events = + [ + new ToolCallStartEvent { ToolCallId = "call_1", ToolCallName = "TestTool", ParentMessageId = "msg1" }, + new ToolCallArgsEvent { ToolCallId = "call_1", Delta = "{\"par" }, + new ToolCallArgsEvent { ToolCallId = "call_1", Delta = "t1\":\"val" }, + new ToolCallArgsEvent { ToolCallId = "call_1", Delta = "ue1\",\"part2" }, + new ToolCallArgsEvent { ToolCallId = "call_1", Delta = "\":\"value2\"}" }, + new ToolCallEndEvent { ToolCallId = "call_1" } + ]; + + // Act + List updates = []; + await foreach (ChatResponseUpdate update in events.ToAsyncEnumerableAsync().AsChatResponseUpdatesAsync(AGUIJsonSerializerContext.Default.Options)) + { + updates.Add(update); + } + + // Assert + FunctionCallContent functionCall = updates + .SelectMany(u => u.Contents) + .OfType() + .Single(); + Assert.Equal("value1", functionCall.Arguments!["part1"]?.ToString()); + Assert.Equal("value2", functionCall.Arguments!["part2"]?.ToString()); + } + + [Fact] + public async Task AsChatResponseUpdatesAsync_WithEmptyToolCallArgs_HandlesGracefullyAsync() + { + // Arrange + List events = + [ + new ToolCallStartEvent { ToolCallId = "call_1", ToolCallName = "NoArgsTool", ParentMessageId = "msg1" }, + new ToolCallArgsEvent { ToolCallId = "call_1", Delta = "" }, + new ToolCallEndEvent { ToolCallId = "call_1" } + ]; + + // Act + List updates = []; + await foreach (ChatResponseUpdate update in events.ToAsyncEnumerableAsync().AsChatResponseUpdatesAsync(AGUIJsonSerializerContext.Default.Options)) + { + updates.Add(update); + } + + // Assert + FunctionCallContent functionCall = updates + .SelectMany(u => u.Contents) + .OfType() + .Single(); + Assert.Equal("call_1", functionCall.CallId); + Assert.Equal("NoArgsTool", functionCall.Name); + Assert.Null(functionCall.Arguments); + } + + [Fact] + public async Task AsChatResponseUpdatesAsync_WithOverlappingToolCalls_ThrowsInvalidOperationExceptionAsync() + { + // Arrange + List events = + [ + new ToolCallStartEvent { ToolCallId = "call_1", ToolCallName = "Tool1", ParentMessageId = "msg1" }, + new ToolCallArgsEvent { ToolCallId = "call_1", Delta = "{}" }, + new ToolCallStartEvent { ToolCallId = "call_2", ToolCallName = "Tool2", ParentMessageId = "msg1" } // Second start before first ends + ]; + + // Act & Assert + await Assert.ThrowsAsync(async () => + { + await foreach (var _ in events.ToAsyncEnumerableAsync().AsChatResponseUpdatesAsync(AGUIJsonSerializerContext.Default.Options)) + { + // Consume stream to trigger exception + } + }); + } + + [Fact] + public async Task AsChatResponseUpdatesAsync_WithMismatchedToolCallId_ThrowsInvalidOperationExceptionAsync() + { + // Arrange + List events = + [ + new ToolCallStartEvent { ToolCallId = "call_1", ToolCallName = "Tool1", ParentMessageId = "msg1" }, + new ToolCallArgsEvent { ToolCallId = "call_2", Delta = "{}" } // Wrong call ID + ]; + + // Act & Assert + await Assert.ThrowsAsync(async () => + { + await foreach (var _ in events.ToAsyncEnumerableAsync().AsChatResponseUpdatesAsync(AGUIJsonSerializerContext.Default.Options)) + { + // Consume stream to trigger exception + } + }); + } + + [Fact] + public async Task AsChatResponseUpdatesAsync_WithMismatchedToolCallEndId_ThrowsInvalidOperationExceptionAsync() + { + // Arrange + List events = + [ + new ToolCallStartEvent { ToolCallId = "call_1", ToolCallName = "Tool1", ParentMessageId = "msg1" }, + new ToolCallArgsEvent { ToolCallId = "call_1", Delta = "{}" }, + new ToolCallEndEvent { ToolCallId = "call_2" } // Wrong call ID + ]; + + // Act & Assert + await Assert.ThrowsAsync(async () => + { + await foreach (var _ in events.ToAsyncEnumerableAsync().AsChatResponseUpdatesAsync(AGUIJsonSerializerContext.Default.Options)) + { + // Consume stream to trigger exception + } + }); + } + + [Fact] + public async Task AsChatResponseUpdatesAsync_WithMultipleSequentialToolCalls_ProcessesAllCorrectlyAsync() + { + // Arrange + List events = + [ + new ToolCallStartEvent { ToolCallId = "call_1", ToolCallName = "Tool1", ParentMessageId = "msg1" }, + new ToolCallArgsEvent { ToolCallId = "call_1", Delta = "{\"arg1\":\"val1\"}" }, + new ToolCallEndEvent { ToolCallId = "call_1" }, + new ToolCallStartEvent { ToolCallId = "call_2", ToolCallName = "Tool2", ParentMessageId = "msg2" }, + new ToolCallArgsEvent { ToolCallId = "call_2", Delta = "{\"arg2\":\"val2\"}" }, + new ToolCallEndEvent { ToolCallId = "call_2" } + ]; + + // Act + List updates = []; + await foreach (ChatResponseUpdate update in events.ToAsyncEnumerableAsync().AsChatResponseUpdatesAsync(AGUIJsonSerializerContext.Default.Options)) + { + updates.Add(update); + } + + // Assert + List functionCalls = updates + .SelectMany(u => u.Contents) + .OfType() + .ToList(); + Assert.Equal(2, functionCalls.Count); + Assert.Equal("call_1", functionCalls[0].CallId); + Assert.Equal("Tool1", functionCalls[0].Name); + Assert.Equal("call_2", functionCalls[1].CallId); + Assert.Equal("Tool2", functionCalls[1].Name); + } + + [Fact] + public async Task AsChatResponseUpdatesAsync_ConvertsStateSnapshotEvent_ToDataContentWithJsonAsync() + { + // Arrange + JsonElement stateSnapshot = JsonSerializer.SerializeToElement(new { counter = 42, status = "active" }); + List events = + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, + new StateSnapshotEvent { Snapshot = stateSnapshot }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } + ]; + + // Act + List updates = []; + await foreach (ChatResponseUpdate update in events.ToAsyncEnumerableAsync().AsChatResponseUpdatesAsync(AGUIJsonSerializerContext.Default.Options)) + { + updates.Add(update); + } + + // Assert + ChatResponseUpdate stateUpdate = updates.First(u => u.Contents.Any(c => c is DataContent)); + Assert.Equal(ChatRole.Assistant, stateUpdate.Role); + Assert.Equal("thread1", stateUpdate.ConversationId); + Assert.Equal("run1", stateUpdate.ResponseId); + + DataContent dataContent = Assert.IsType(stateUpdate.Contents[0]); + Assert.Equal("application/json", dataContent.MediaType); + + // Verify the JSON content + string jsonText = System.Text.Encoding.UTF8.GetString(dataContent.Data.ToArray()); + JsonElement deserializedState = JsonElement.Parse(jsonText); + Assert.Equal(42, deserializedState.GetProperty("counter").GetInt32()); + Assert.Equal("active", deserializedState.GetProperty("status").GetString()); + + // Verify additional properties + Assert.NotNull(stateUpdate.AdditionalProperties); + Assert.True((bool)stateUpdate.AdditionalProperties["is_state_snapshot"]!); + } + + [Fact] + public async Task AsChatResponseUpdatesAsync_WithNullStateSnapshot_DoesNotEmitUpdateAsync() + { + // Arrange + List events = + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, + new StateSnapshotEvent { Snapshot = null }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } + ]; + + // Act + List updates = []; + await foreach (ChatResponseUpdate update in events.ToAsyncEnumerableAsync().AsChatResponseUpdatesAsync(AGUIJsonSerializerContext.Default.Options)) + { + updates.Add(update); + } + + // Assert + Assert.DoesNotContain(updates, u => u.Contents.Any(c => c is DataContent)); + } + + [Fact] + public async Task AsChatResponseUpdatesAsync_WithEmptyObjectStateSnapshot_EmitsDataContentAsync() + { + // Arrange + JsonElement emptyState = JsonSerializer.SerializeToElement(new { }); + List events = + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, + new StateSnapshotEvent { Snapshot = emptyState }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } + ]; + + // Act + List updates = []; + await foreach (ChatResponseUpdate update in events.ToAsyncEnumerableAsync().AsChatResponseUpdatesAsync(AGUIJsonSerializerContext.Default.Options)) + { + updates.Add(update); + } + + // Assert + ChatResponseUpdate stateUpdate = updates.First(u => u.Contents.Any(c => c is DataContent)); + DataContent dataContent = Assert.IsType(stateUpdate.Contents[0]); + string jsonText = System.Text.Encoding.UTF8.GetString(dataContent.Data.ToArray()); + Assert.Equal("{}", jsonText); + } + + [Fact] + public async Task AsChatResponseUpdatesAsync_WithComplexStateSnapshot_PreservesJsonStructureAsync() + { + // Arrange + var complexState = new + { + user = new { name = "Alice", age = 30 }, + items = new[] { "item1", "item2", "item3" }, + metadata = new { timestamp = "2024-01-01T00:00:00Z", version = 2 } + }; + JsonElement stateSnapshot = JsonSerializer.SerializeToElement(complexState); + List events = + [ + new StateSnapshotEvent { Snapshot = stateSnapshot } + ]; + + // Act + List updates = []; + await foreach (ChatResponseUpdate update in events.ToAsyncEnumerableAsync().AsChatResponseUpdatesAsync(AGUIJsonSerializerContext.Default.Options)) + { + updates.Add(update); + } + + // Assert + ChatResponseUpdate stateUpdate = updates.First(); + DataContent dataContent = Assert.IsType(stateUpdate.Contents[0]); + string jsonText = System.Text.Encoding.UTF8.GetString(dataContent.Data.ToArray()); + JsonElement roundTrippedState = JsonElement.Parse(jsonText); + + Assert.Equal("Alice", roundTrippedState.GetProperty("user").GetProperty("name").GetString()); + Assert.Equal(30, roundTrippedState.GetProperty("user").GetProperty("age").GetInt32()); + Assert.Equal(3, roundTrippedState.GetProperty("items").GetArrayLength()); + Assert.Equal("item1", roundTrippedState.GetProperty("items")[0].GetString()); + } + + [Fact] + public async Task AsChatResponseUpdatesAsync_WithStateSnapshotAndTextMessages_EmitsBothAsync() + { + // Arrange + JsonElement state = JsonSerializer.SerializeToElement(new { step = 1 }); + List events = + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, + new TextMessageStartEvent { MessageId = "msg1", Role = AGUIRoles.Assistant }, + new TextMessageContentEvent { MessageId = "msg1", Delta = "Processing..." }, + new TextMessageEndEvent { MessageId = "msg1" }, + new StateSnapshotEvent { Snapshot = state }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } + ]; + + // Act + List updates = []; + await foreach (ChatResponseUpdate update in events.ToAsyncEnumerableAsync().AsChatResponseUpdatesAsync(AGUIJsonSerializerContext.Default.Options)) + { + updates.Add(update); + } + + // Assert + Assert.Contains(updates, u => u.Contents.Any(c => c is TextContent)); + Assert.Contains(updates, u => u.Contents.Any(c => c is DataContent)); + } + + #region State Delta Tests + + [Fact] + public async Task AsChatResponseUpdatesAsync_ConvertsStateDeltaEvent_ToDataContentWithJsonPatchAsync() + { + // Arrange - Create JSON Patch operations (RFC 6902) + JsonElement stateDelta = JsonSerializer.SerializeToElement(new object[] + { + new { op = "replace", path = "/counter", value = 43 }, + new { op = "add", path = "/newField", value = "test" } + }); + List events = + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, + new StateDeltaEvent { Delta = stateDelta }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } + ]; + + // Act + List updates = []; + await foreach (ChatResponseUpdate update in events.ToAsyncEnumerableAsync().AsChatResponseUpdatesAsync(AGUIJsonSerializerContext.Default.Options)) + { + updates.Add(update); + } + + // Assert + ChatResponseUpdate deltaUpdate = updates.First(u => u.Contents.Any(c => c is DataContent dc && dc.MediaType == "application/json-patch+json")); + Assert.Equal(ChatRole.Assistant, deltaUpdate.Role); + Assert.Equal("thread1", deltaUpdate.ConversationId); + Assert.Equal("run1", deltaUpdate.ResponseId); + + DataContent dataContent = Assert.IsType(deltaUpdate.Contents[0]); + Assert.Equal("application/json-patch+json", dataContent.MediaType); + + // Verify the JSON Patch content + string jsonText = System.Text.Encoding.UTF8.GetString(dataContent.Data.ToArray()); + JsonElement deserializedDelta = JsonElement.Parse(jsonText); + Assert.Equal(JsonValueKind.Array, deserializedDelta.ValueKind); + Assert.Equal(2, deserializedDelta.GetArrayLength()); + + // Verify first operation + JsonElement firstOp = deserializedDelta[0]; + Assert.Equal("replace", firstOp.GetProperty("op").GetString()); + Assert.Equal("/counter", firstOp.GetProperty("path").GetString()); + Assert.Equal(43, firstOp.GetProperty("value").GetInt32()); + + // Verify second operation + JsonElement secondOp = deserializedDelta[1]; + Assert.Equal("add", secondOp.GetProperty("op").GetString()); + Assert.Equal("/newField", secondOp.GetProperty("path").GetString()); + Assert.Equal("test", secondOp.GetProperty("value").GetString()); + + // Verify additional properties + Assert.NotNull(deltaUpdate.AdditionalProperties); + Assert.True((bool)deltaUpdate.AdditionalProperties["is_state_delta"]!); + } + + [Fact] + public async Task AsChatResponseUpdatesAsync_WithNullStateDelta_DoesNotEmitUpdateAsync() + { + // Arrange + List events = + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, + new StateDeltaEvent { Delta = null }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } + ]; + + // Act + List updates = []; + await foreach (ChatResponseUpdate update in events.ToAsyncEnumerableAsync().AsChatResponseUpdatesAsync(AGUIJsonSerializerContext.Default.Options)) + { + updates.Add(update); + } + + // Assert - Only run started and finished should be present + Assert.Equal(2, updates.Count); + Assert.IsType(updates[0]); // Run started + Assert.IsType(updates[1]); // Run finished + Assert.DoesNotContain(updates, u => u.Contents.Any(c => c is DataContent)); + } + + [Fact] + public async Task AsChatResponseUpdatesAsync_WithEmptyStateDelta_EmitsUpdateAsync() + { + // Arrange - Empty JSON Patch array is valid + JsonElement emptyDelta = JsonSerializer.SerializeToElement(Array.Empty()); + List events = + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, + new StateDeltaEvent { Delta = emptyDelta }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } + ]; + + // Act + List updates = []; + await foreach (ChatResponseUpdate update in events.ToAsyncEnumerableAsync().AsChatResponseUpdatesAsync(AGUIJsonSerializerContext.Default.Options)) + { + updates.Add(update); + } + + // Assert + Assert.Contains(updates, u => u.Contents.Any(c => c is DataContent dc && dc.MediaType == "application/json-patch+json")); + } + + [Fact] + public async Task AsChatResponseUpdatesAsync_WithMultipleStateDeltaEvents_ConvertsAllAsync() + { + // Arrange + JsonElement delta1 = JsonSerializer.SerializeToElement(new[] { new { op = "replace", path = "/counter", value = 1 } }); + JsonElement delta2 = JsonSerializer.SerializeToElement(new[] { new { op = "replace", path = "/counter", value = 2 } }); + JsonElement delta3 = JsonSerializer.SerializeToElement(new[] { new { op = "replace", path = "/counter", value = 3 } }); + + List events = + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, + new StateDeltaEvent { Delta = delta1 }, + new StateDeltaEvent { Delta = delta2 }, + new StateDeltaEvent { Delta = delta3 }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } + ]; + + // Act + List updates = []; + await foreach (ChatResponseUpdate update in events.ToAsyncEnumerableAsync().AsChatResponseUpdatesAsync(AGUIJsonSerializerContext.Default.Options)) + { + updates.Add(update); + } + + // Assert + var deltaUpdates = updates.Where(u => u.Contents.Any(c => c is DataContent dc && dc.MediaType == "application/json-patch+json")).ToList(); + Assert.Equal(3, deltaUpdates.Count); + } + + [Fact] + public async Task AsAGUIEventStreamAsync_ConvertsDataContentWithJsonPatch_ToStateDeltaEventAsync() + { + // Arrange - Create a ChatResponseUpdate with JSON Patch DataContent + JsonElement patchOps = JsonSerializer.SerializeToElement(new object[] + { + new { op = "remove", path = "/oldField" }, + new { op = "add", path = "/newField", value = "newValue" } + }); + byte[] jsonBytes = JsonSerializer.SerializeToUtf8Bytes(patchOps); + DataContent dataContent = new(jsonBytes, "application/json-patch+json"); + + List updates = + [ + new ChatResponseUpdate(ChatRole.Assistant, [dataContent]) + { + MessageId = "msg1" + } + ]; + + // Act + List outputEvents = []; + await foreach (BaseEvent evt in updates.ToAsyncEnumerableAsync().AsAGUIEventStreamAsync("thread1", "run1", AGUIJsonSerializerContext.Default.Options)) + { + outputEvents.Add(evt); + } + + // Assert + StateDeltaEvent? deltaEvent = outputEvents.OfType().FirstOrDefault(); + Assert.NotNull(deltaEvent); + Assert.NotNull(deltaEvent.Delta); + Assert.Equal(JsonValueKind.Array, deltaEvent.Delta.Value.ValueKind); + + // Verify patch operations + JsonElement delta = deltaEvent.Delta.Value; + Assert.Equal(2, delta.GetArrayLength()); + Assert.Equal("remove", delta[0].GetProperty("op").GetString()); + Assert.Equal("/oldField", delta[0].GetProperty("path").GetString()); + Assert.Equal("add", delta[1].GetProperty("op").GetString()); + Assert.Equal("/newField", delta[1].GetProperty("path").GetString()); + } + + [Fact] + public async Task AsAGUIEventStreamAsync_WithBothSnapshotAndDelta_EmitsBothEventsAsync() + { + // Arrange + JsonElement snapshot = JsonSerializer.SerializeToElement(new { counter = 0 }); + byte[] snapshotBytes = JsonSerializer.SerializeToUtf8Bytes(snapshot); + DataContent snapshotContent = new(snapshotBytes, "application/json"); + + JsonElement delta = JsonSerializer.SerializeToElement(new[] { new { op = "replace", path = "/counter", value = 1 } }); + byte[] deltaBytes = JsonSerializer.SerializeToUtf8Bytes(delta); + DataContent deltaContent = new(deltaBytes, "application/json-patch+json"); + + List updates = + [ + new ChatResponseUpdate(ChatRole.Assistant, [snapshotContent]) { MessageId = "msg1" }, + new ChatResponseUpdate(ChatRole.Assistant, [deltaContent]) { MessageId = "msg2" } + ]; + + // Act + List outputEvents = []; + await foreach (BaseEvent evt in updates.ToAsyncEnumerableAsync().AsAGUIEventStreamAsync("thread1", "run1", AGUIJsonSerializerContext.Default.Options)) + { + outputEvents.Add(evt); + } + + // Assert + Assert.Contains(outputEvents, e => e is StateSnapshotEvent); + Assert.Contains(outputEvents, e => e is StateDeltaEvent); + } + + [Fact] + public async Task StateDeltaEvent_RoundTrip_PreservesJsonPatchOperationsAsync() + { + // Arrange - Create complex JSON Patch with various operations + JsonElement originalDelta = JsonSerializer.SerializeToElement(new object[] + { + new { op = "add", path = "/user/email", value = "test@example.com" }, + new { op = "remove", path = "/user/tempData" }, + new { op = "replace", path = "/user/lastLogin", value = "2025-11-09T12:00:00Z" }, + new { op = "move", from = "/user/oldAddress", path = "/user/previousAddress" }, + new { op = "copy", from = "/user/name", path = "/user/displayName" }, + new { op = "test", path = "/user/version", value = 2 } + }); + + List events = + [ + new RunStartedEvent { ThreadId = "thread1", RunId = "run1" }, + new StateDeltaEvent { Delta = originalDelta }, + new RunFinishedEvent { ThreadId = "thread1", RunId = "run1" } + ]; + + // Act - Convert to ChatResponseUpdate and back to events + List updates = []; + await foreach (ChatResponseUpdate update in events.ToAsyncEnumerableAsync().AsChatResponseUpdatesAsync(AGUIJsonSerializerContext.Default.Options)) + { + updates.Add(update); + } + + List roundTripEvents = []; + await foreach (BaseEvent evt in updates.ToAsyncEnumerableAsync().AsAGUIEventStreamAsync("thread1", "run1", AGUIJsonSerializerContext.Default.Options)) + { + roundTripEvents.Add(evt); + } + + // Assert + StateDeltaEvent? roundTripDelta = roundTripEvents.OfType().FirstOrDefault(); + Assert.NotNull(roundTripDelta); + Assert.NotNull(roundTripDelta.Delta); + + JsonElement delta = roundTripDelta.Delta.Value; + Assert.Equal(6, delta.GetArrayLength()); + + // Verify each operation type + Assert.Equal("add", delta[0].GetProperty("op").GetString()); + Assert.Equal("remove", delta[1].GetProperty("op").GetString()); + Assert.Equal("replace", delta[2].GetProperty("op").GetString()); + Assert.Equal("move", delta[3].GetProperty("op").GetString()); + Assert.Equal("copy", delta[4].GetProperty("op").GetString()); + Assert.Equal("test", delta[5].GetProperty("op").GetString()); + } + + #endregion State Delta Tests +} diff --git a/dotnet/tests/Microsoft.Agents.AI.AGUI.UnitTests/Microsoft.Agents.AI.AGUI.UnitTests.csproj b/dotnet/tests/Microsoft.Agents.AI.AGUI.UnitTests/Microsoft.Agents.AI.AGUI.UnitTests.csproj index 276af004d8..0dab0aa9e4 100644 --- a/dotnet/tests/Microsoft.Agents.AI.AGUI.UnitTests/Microsoft.Agents.AI.AGUI.UnitTests.csproj +++ b/dotnet/tests/Microsoft.Agents.AI.AGUI.UnitTests/Microsoft.Agents.AI.AGUI.UnitTests.csproj @@ -1,17 +1,11 @@ - - $(ProjectsTargetFrameworks) - - - - - + diff --git a/dotnet/tests/Microsoft.Agents.AI.Abstractions.UnitTests/AIAgentTests.cs b/dotnet/tests/Microsoft.Agents.AI.Abstractions.UnitTests/AIAgentTests.cs index bfa14a89d4..e3bda2081a 100644 --- a/dotnet/tests/Microsoft.Agents.AI.Abstractions.UnitTests/AIAgentTests.cs +++ b/dotnet/tests/Microsoft.Agents.AI.Abstractions.UnitTests/AIAgentTests.cs @@ -8,7 +8,6 @@ using System.Threading.Tasks; using Microsoft.Extensions.AI; using Moq; -using Moq.Protected; namespace Microsoft.Agents.AI.Abstractions.UnitTests; @@ -215,26 +214,29 @@ public async Task InvokeStreamingWithSingleMessageCallsMockedInvokeWithMessageIn [Fact] public void ValidateAgentIDIsIdempotent() { + // Arrange var agent = new MockAgent(); + // Act string id = agent.Id; + + // Assert Assert.NotNull(id); Assert.Equal(id, agent.Id); } [Fact] - public async Task NotifyThreadOfNewMessagesNotifiesThreadAsync() + public void ValidateAgentIDCanBeProvidedByDerivedAgentClass() { - var cancellationToken = default(CancellationToken); - - var messages = new[] { new ChatMessage(ChatRole.User, "msg1"), new ChatMessage(ChatRole.User, "msg2") }; - - var threadMock = new Mock { CallBase = true }; - threadMock.SetupAllProperties(); + // Arrange + var agent = new MockAgent(id: "test-agent-id"); - await MockAgent.NotifyThreadOfNewMessagesAsync(threadMock.Object, messages, cancellationToken); + // Act + string id = agent.Id; - threadMock.Protected().Verify("MessagesReceivedAsync", Times.Once(), messages, cancellationToken); + // Assert + Assert.NotNull(id); + Assert.Equal("test-agent-id", id); } #region GetService Method Tests @@ -360,8 +362,12 @@ public abstract class TestAgentThread : AgentThread; private sealed class MockAgent : AIAgent { - public static new Task NotifyThreadOfNewMessagesAsync(AgentThread thread, IEnumerable messages, CancellationToken cancellationToken) => - AIAgent.NotifyThreadOfNewMessagesAsync(thread, messages, cancellationToken); + public MockAgent(string? id = null) + { + this.IdCore = id; + } + + protected override string? IdCore { get; } public override AgentThread GetNewThread() => throw new NotImplementedException(); diff --git a/dotnet/tests/Microsoft.Agents.AI.Abstractions.UnitTests/AIContextProviderTests.cs b/dotnet/tests/Microsoft.Agents.AI.Abstractions.UnitTests/AIContextProviderTests.cs index 0b8f41f1bb..b287c8b304 100644 --- a/dotnet/tests/Microsoft.Agents.AI.Abstractions.UnitTests/AIContextProviderTests.cs +++ b/dotnet/tests/Microsoft.Agents.AI.Abstractions.UnitTests/AIContextProviderTests.cs @@ -2,7 +2,6 @@ using System; using System.Collections.ObjectModel; -using System.Text.Json; using System.Threading; using System.Threading.Tasks; using Microsoft.Extensions.AI; @@ -162,10 +161,5 @@ public override ValueTask InvokingAsync(InvokingContext context, Canc { return default; } - - public override JsonElement Serialize(JsonSerializerOptions? jsonSerializerOptions = null) - { - return base.Serialize(jsonSerializerOptions); - } } } diff --git a/dotnet/tests/Microsoft.Agents.AI.Abstractions.UnitTests/AgentRunOptionsTests.cs b/dotnet/tests/Microsoft.Agents.AI.Abstractions.UnitTests/AgentRunOptionsTests.cs index 40901a4969..7460ea4623 100644 --- a/dotnet/tests/Microsoft.Agents.AI.Abstractions.UnitTests/AgentRunOptionsTests.cs +++ b/dotnet/tests/Microsoft.Agents.AI.Abstractions.UnitTests/AgentRunOptionsTests.cs @@ -17,8 +17,13 @@ public void CloningConstructorCopiesProperties() // Arrange var options = new AgentRunOptions { - ContinuationToken = new object(), - AllowBackgroundResponses = true + ContinuationToken = ResponseContinuationToken.FromBytes(new byte[] { 1, 2, 3 }), + AllowBackgroundResponses = true, + AdditionalProperties = new AdditionalPropertiesDictionary + { + ["key1"] = "value1", + ["key2"] = 42 + } }; // Act @@ -28,6 +33,10 @@ public void CloningConstructorCopiesProperties() Assert.NotNull(clone); Assert.Same(options.ContinuationToken, clone.ContinuationToken); Assert.Equal(options.AllowBackgroundResponses, clone.AllowBackgroundResponses); + Assert.NotNull(clone.AdditionalProperties); + Assert.NotSame(options.AdditionalProperties, clone.AdditionalProperties); + Assert.Equal("value1", clone.AdditionalProperties["key1"]); + Assert.Equal(42, clone.AdditionalProperties["key2"]); } [Fact] @@ -42,7 +51,12 @@ public void JsonSerializationRoundtrips() var options = new AgentRunOptions { ContinuationToken = ResponseContinuationToken.FromBytes(new byte[] { 1, 2, 3 }), - AllowBackgroundResponses = true + AllowBackgroundResponses = true, + AdditionalProperties = new AdditionalPropertiesDictionary + { + ["key1"] = "value1", + ["key2"] = 42 + } }; // Act @@ -54,5 +68,13 @@ public void JsonSerializationRoundtrips() Assert.NotNull(deserialized); Assert.Equivalent(ResponseContinuationToken.FromBytes(new byte[] { 1, 2, 3 }), deserialized!.ContinuationToken); Assert.Equal(options.AllowBackgroundResponses, deserialized.AllowBackgroundResponses); + Assert.NotNull(deserialized.AdditionalProperties); + Assert.Equal(2, deserialized.AdditionalProperties.Count); + Assert.True(deserialized.AdditionalProperties.TryGetValue("key1", out object? value1)); + Assert.IsType(value1); + Assert.Equal("value1", ((JsonElement)value1!).GetString()); + Assert.True(deserialized.AdditionalProperties.TryGetValue("key2", out object? value2)); + Assert.IsType(value2); + Assert.Equal(42, ((JsonElement)value2!).GetInt32()); } } diff --git a/dotnet/tests/Microsoft.Agents.AI.Abstractions.UnitTests/AgentRunResponseUpdateTests.cs b/dotnet/tests/Microsoft.Agents.AI.Abstractions.UnitTests/AgentRunResponseUpdateTests.cs index 42d3fdf199..32b7acd673 100644 --- a/dotnet/tests/Microsoft.Agents.AI.Abstractions.UnitTests/AgentRunResponseUpdateTests.cs +++ b/dotnet/tests/Microsoft.Agents.AI.Abstractions.UnitTests/AgentRunResponseUpdateTests.cs @@ -42,7 +42,7 @@ public void ConstructorWithChatResponseUpdateRoundtrips() RawRepresentation = new object(), ResponseId = "responseId", Role = ChatRole.Assistant, - ContinuationToken = new object(), + ContinuationToken = ResponseContinuationToken.FromBytes(new byte[] { 1, 2, 3 }), }; AgentRunResponseUpdate response = new(chatResponseUpdate); diff --git a/dotnet/tests/Microsoft.Agents.AI.Abstractions.UnitTests/AgentThreadTests.cs b/dotnet/tests/Microsoft.Agents.AI.Abstractions.UnitTests/AgentThreadTests.cs index 4d7c4ad219..e75cb4caa1 100644 --- a/dotnet/tests/Microsoft.Agents.AI.Abstractions.UnitTests/AgentThreadTests.cs +++ b/dotnet/tests/Microsoft.Agents.AI.Abstractions.UnitTests/AgentThreadTests.cs @@ -1,8 +1,6 @@ // Copyright (c) Microsoft. All rights reserved. using System; -using System.Collections.Generic; -using Microsoft.Extensions.AI; #pragma warning disable CA1861 // Avoid constant arrays as arguments @@ -21,15 +19,6 @@ public void Serialize_ReturnsDefaultJsonElement() Assert.Equal(default, result); } - [Fact] - public void MessagesReceivedAsync_ReturnsCompletedTask() - { - var thread = new TestAgentThread(); - var messages = new List { new(ChatRole.User, "hello") }; - var result = thread.MessagesReceivedAsync(messages); - Assert.True(result.IsCompleted); - } - #region GetService Method Tests /// diff --git a/dotnet/tests/Microsoft.Agents.AI.Abstractions.UnitTests/DelegatingAIAgentTests.cs b/dotnet/tests/Microsoft.Agents.AI.Abstractions.UnitTests/DelegatingAIAgentTests.cs index 4dca99a77c..50271b7eee 100644 --- a/dotnet/tests/Microsoft.Agents.AI.Abstractions.UnitTests/DelegatingAIAgentTests.cs +++ b/dotnet/tests/Microsoft.Agents.AI.Abstractions.UnitTests/DelegatingAIAgentTests.cs @@ -6,6 +6,7 @@ using System.Threading.Tasks; using Microsoft.Extensions.AI; using Moq; +using Moq.Protected; namespace Microsoft.Agents.AI.Abstractions.UnitTests; @@ -31,7 +32,7 @@ public DelegatingAIAgentTests() this._testThread = new TestAgentThread(); // Setup inner agent mock - this._innerAgentMock.Setup(x => x.Id).Returns("test-agent-id"); + this._innerAgentMock.Protected().SetupGet("IdCore").Returns("test-agent-id"); this._innerAgentMock.Setup(x => x.Name).Returns("Test Agent"); this._innerAgentMock.Setup(x => x.Description).Returns("Test Description"); this._innerAgentMock.Setup(x => x.GetNewThread()).Returns(this._testThread); @@ -93,7 +94,7 @@ public void Id_DelegatesToInnerAgent() // Assert Assert.Equal("test-agent-id", id); - this._innerAgentMock.Verify(x => x.Id, Times.Once); + this._innerAgentMock.Protected().VerifyGet("IdCore", Times.Once()); } /// diff --git a/dotnet/tests/Microsoft.Agents.AI.Abstractions.UnitTests/InMemoryChatMessageStoreTests.cs b/dotnet/tests/Microsoft.Agents.AI.Abstractions.UnitTests/InMemoryChatMessageStoreTests.cs index 4c793d17f4..824fb62f6d 100644 --- a/dotnet/tests/Microsoft.Agents.AI.Abstractions.UnitTests/InMemoryChatMessageStoreTests.cs +++ b/dotnet/tests/Microsoft.Agents.AI.Abstractions.UnitTests/InMemoryChatMessageStoreTests.cs @@ -3,7 +3,9 @@ using System; using System.Collections.Generic; using System.Linq; +using System.Text.Encodings.Web; using System.Text.Json; +using System.Text.Json.Serialization.Metadata; using System.Threading; using System.Threading.Tasks; using Microsoft.Extensions.AI; @@ -114,6 +116,29 @@ public async Task SerializeAndDeserializeConstructorRoundtripsAsync() Assert.Equal("B", newStore[1].Text); } + [Fact] + public async Task SerializeAndDeserializeConstructorRoundtripsWithCustomAIContentAsync() + { + JsonSerializerOptions options = new(TestJsonSerializerContext.Default.Options) + { + TypeInfoResolver = JsonTypeInfoResolver.Combine(AgentAbstractionsJsonUtilities.DefaultOptions.TypeInfoResolver, TestJsonSerializerContext.Default), + Encoder = JavaScriptEncoder.UnsafeRelaxedJsonEscaping, + }; + options.AddAIContentType(typeDiscriminatorId: "testContent"); + + var store = new InMemoryChatMessageStore + { + new ChatMessage(ChatRole.User, [new TestAIContent("foo data")]), + }; + + var jsonElement = store.Serialize(options); + var newStore = new InMemoryChatMessageStore(jsonElement, options); + + Assert.Single(newStore); + var actualTestAIContent = Assert.IsType(newStore[0].Contents[0]); + Assert.Equal("foo data", actualTestAIContent.TestData); + } + [Fact] public async Task SerializeAndDeserializeWorksWithExperimentalContentTypesAsync() { @@ -558,4 +583,9 @@ public async Task GetMessagesAsync_WithReducer_ButWrongTrigger_DoesNotInvokeRedu Assert.Equal("Hello", result[0].Text); reducerMock.Verify(r => r.ReduceAsync(It.IsAny>(), It.IsAny()), Times.Never); } + + public class TestAIContent(string testData) : AIContent + { + public string TestData => testData; + } } diff --git a/dotnet/tests/Microsoft.Agents.AI.Abstractions.UnitTests/Microsoft.Agents.AI.Abstractions.UnitTests.csproj b/dotnet/tests/Microsoft.Agents.AI.Abstractions.UnitTests/Microsoft.Agents.AI.Abstractions.UnitTests.csproj index b7c5412a53..1e5db6ed29 100644 --- a/dotnet/tests/Microsoft.Agents.AI.Abstractions.UnitTests/Microsoft.Agents.AI.Abstractions.UnitTests.csproj +++ b/dotnet/tests/Microsoft.Agents.AI.Abstractions.UnitTests/Microsoft.Agents.AI.Abstractions.UnitTests.csproj @@ -1,7 +1,6 @@ - $(ProjectsTargetFrameworks) $(NoWarn);MEAI001 @@ -13,9 +12,8 @@ - - - + + diff --git a/dotnet/tests/Microsoft.Agents.AI.Abstractions.UnitTests/ServiceIdAgentThreadTests.cs b/dotnet/tests/Microsoft.Agents.AI.Abstractions.UnitTests/ServiceIdAgentThreadTests.cs index e451359c23..1da79344d4 100644 --- a/dotnet/tests/Microsoft.Agents.AI.Abstractions.UnitTests/ServiceIdAgentThreadTests.cs +++ b/dotnet/tests/Microsoft.Agents.AI.Abstractions.UnitTests/ServiceIdAgentThreadTests.cs @@ -115,7 +115,5 @@ public TestServiceIdAgentThread(JsonElement serializedThreadState) : base(serial } // Helper class to represent empty objects - internal sealed class EmptyObject - { - } + internal sealed class EmptyObject; } diff --git a/dotnet/tests/Microsoft.Agents.AI.Abstractions.UnitTests/TestJsonSerializerContext.cs b/dotnet/tests/Microsoft.Agents.AI.Abstractions.UnitTests/TestJsonSerializerContext.cs index b7c553d348..ec343504ab 100644 --- a/dotnet/tests/Microsoft.Agents.AI.Abstractions.UnitTests/TestJsonSerializerContext.cs +++ b/dotnet/tests/Microsoft.Agents.AI.Abstractions.UnitTests/TestJsonSerializerContext.cs @@ -22,4 +22,5 @@ namespace Microsoft.Agents.AI.Abstractions.UnitTests; [JsonSerializable(typeof(InMemoryAgentThread.InMemoryAgentThreadState))] [JsonSerializable(typeof(ServiceIdAgentThread.ServiceIdAgentThreadState))] [JsonSerializable(typeof(ServiceIdAgentThreadTests.EmptyObject))] +[JsonSerializable(typeof(InMemoryChatMessageStoreTests.TestAIContent))] internal sealed partial class TestJsonSerializerContext : JsonSerializerContext; diff --git a/dotnet/tests/Microsoft.Agents.AI.Anthropic.UnitTests/Extensions/AnthropicBetaServiceExtensionsTests.cs b/dotnet/tests/Microsoft.Agents.AI.Anthropic.UnitTests/Extensions/AnthropicBetaServiceExtensionsTests.cs new file mode 100644 index 0000000000..400bcf5456 --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.Anthropic.UnitTests/Extensions/AnthropicBetaServiceExtensionsTests.cs @@ -0,0 +1,290 @@ +// Copyright (c) Microsoft. All rights reserved. + +#pragma warning disable IDE0052 // Remove unread private members + +using System; +using System.Collections.Generic; +using System.Net.Http; +using System.Runtime.CompilerServices; +using System.Threading; +using System.Threading.Tasks; +using Anthropic; +using Anthropic.Core; +using Anthropic.Services; +using Microsoft.Extensions.AI; +using Moq; +using IBetaMessageService = Anthropic.Services.Beta.IMessageService; +using IMessageService = Anthropic.Services.IMessageService; + +namespace Microsoft.Agents.AI.Anthropic.UnitTests.Extensions; + +/// +/// Unit tests for the AnthropicClientExtensions class. +/// +public sealed class AnthropicBetaServiceExtensionsTests +{ + /// + /// Verify that CreateAIAgent with clientFactory parameter correctly applies the factory. + /// + [Fact] + public void CreateAIAgent_WithClientFactory_AppliesFactoryCorrectly() + { + // Arrange + var chatClient = new TestAnthropicChatClient(); + var testChatClient = new TestChatClient(chatClient.Beta.AsIChatClient()); + + // Act + var agent = chatClient.Beta.CreateAIAgent( + model: "test-model", + instructions: "Test instructions", + name: "Test Agent", + description: "Test description", + clientFactory: (innerClient) => testChatClient); + + // Assert + Assert.NotNull(agent); + Assert.Equal("Test Agent", agent.Name); + Assert.Equal("Test description", agent.Description); + + // Verify that the custom chat client can be retrieved from the agent's service collection + var retrievedTestClient = agent.GetService(); + Assert.NotNull(retrievedTestClient); + Assert.Same(testChatClient, retrievedTestClient); + } + + /// + /// Verify that CreateAIAgent with clientFactory using AsBuilder pattern works correctly. + /// + [Fact] + public void CreateAIAgent_WithClientFactoryUsingAsBuilder_AppliesFactoryCorrectly() + { + // Arrange + var chatClient = new TestAnthropicChatClient(); + TestChatClient? testChatClient = null; + + // Act + var agent = chatClient.Beta.CreateAIAgent( + model: "test-model", + instructions: "Test instructions", + clientFactory: (innerClient) => + innerClient.AsBuilder().Use((innerClient) => testChatClient = new TestChatClient(innerClient)).Build()); + + // Assert + Assert.NotNull(agent); + + // Verify that the custom chat client can be retrieved from the agent's service collection + var retrievedTestClient = agent.GetService(); + Assert.NotNull(retrievedTestClient); + Assert.Same(testChatClient, retrievedTestClient); + } + + /// + /// Verify that CreateAIAgent with options and clientFactory parameter correctly applies the factory. + /// + [Fact] + public void CreateAIAgent_WithOptionsAndClientFactory_AppliesFactoryCorrectly() + { + // Arrange + var chatClient = new TestAnthropicChatClient(); + var testChatClient = new TestChatClient(chatClient.Beta.AsIChatClient()); + var options = new ChatClientAgentOptions + { + Name = "Test Agent", + Description = "Test description", + ChatOptions = new() { Instructions = "Test instructions" } + }; + + // Act + var agent = chatClient.Beta.CreateAIAgent( + options, + clientFactory: (innerClient) => testChatClient); + + // Assert + Assert.NotNull(agent); + Assert.Equal("Test Agent", agent.Name); + Assert.Equal("Test description", agent.Description); + + // Verify that the custom chat client can be retrieved from the agent's service collection + var retrievedTestClient = agent.GetService(); + Assert.NotNull(retrievedTestClient); + Assert.Same(testChatClient, retrievedTestClient); + } + + /// + /// Verify that CreateAIAgent without clientFactory works normally. + /// + [Fact] + public void CreateAIAgent_WithoutClientFactory_WorksNormally() + { + // Arrange + var chatClient = new TestAnthropicChatClient(); + + // Act + var agent = chatClient.Beta.CreateAIAgent( + model: "test-model", + instructions: "Test instructions", + name: "Test Agent"); + + // Assert + Assert.NotNull(agent); + Assert.Equal("Test Agent", agent.Name); + + // Verify that no TestChatClient is available since no factory was provided + var retrievedTestClient = agent.GetService(); + Assert.Null(retrievedTestClient); + } + + /// + /// Verify that CreateAIAgent with null clientFactory works normally. + /// + [Fact] + public void CreateAIAgent_WithNullClientFactory_WorksNormally() + { + // Arrange + var chatClient = new TestAnthropicChatClient(); + + // Act + var agent = chatClient.Beta.CreateAIAgent( + model: "test-model", + instructions: "Test instructions", + name: "Test Agent", + clientFactory: null); + + // Assert + Assert.NotNull(agent); + Assert.Equal("Test Agent", agent.Name); + + // Verify that no TestChatClient is available since no factory was provided + var retrievedTestClient = agent.GetService(); + Assert.Null(retrievedTestClient); + } + + /// + /// Verify that CreateAIAgent throws ArgumentNullException when client is null. + /// + [Fact] + public void CreateAIAgent_WithNullClient_ThrowsArgumentNullException() + { + // Act & Assert + var exception = Assert.Throws(() => + ((IBetaService)null!).CreateAIAgent("test-model")); + + Assert.Equal("betaService", exception.ParamName); + } + + /// + /// Verify that CreateAIAgent with options throws ArgumentNullException when options is null. + /// + [Fact] + public void CreateAIAgent_WithNullOptions_ThrowsArgumentNullException() + { + // Arrange + var chatClient = new TestAnthropicChatClient(); + + // Act & Assert + var exception = Assert.Throws(() => + chatClient.Beta.CreateAIAgent((ChatClientAgentOptions)null!)); + + Assert.Equal("options", exception.ParamName); + } + + /// + /// Test custom chat client that can be used to verify clientFactory functionality. + /// + private sealed class TestChatClient : IChatClient + { + private readonly IChatClient _innerClient; + + public TestChatClient(IChatClient innerClient) + { + this._innerClient = innerClient; + } + + public Task GetResponseAsync(IEnumerable messages, ChatOptions? options = null, CancellationToken cancellationToken = default) + => this._innerClient.GetResponseAsync(messages, options, cancellationToken); + + public async IAsyncEnumerable GetStreamingResponseAsync( + IEnumerable messages, ChatOptions? options = null, [EnumeratorCancellation] CancellationToken cancellationToken = default) + { + await foreach (var update in this._innerClient.GetStreamingResponseAsync(messages, options, cancellationToken)) + { + yield return update; + } + } + + public object? GetService(Type serviceType, object? serviceKey = null) + { + // Return this instance when requested + if (serviceType == typeof(TestChatClient)) + { + return this; + } + + return this._innerClient.GetService(serviceType, serviceKey); + } + + public void Dispose() => this._innerClient.Dispose(); + } + + /// + /// Creates a test ChatClient implementation for testing. + /// + private sealed class TestAnthropicChatClient : IAnthropicClient + { + public TestAnthropicChatClient() + { + this.BetaService = new TestBetaService(this); + } + + public HttpClient HttpClient { get => throw new NotImplementedException(); init => throw new NotImplementedException(); } + public Uri BaseUrl { get => new("http://localhost"); init => throw new NotImplementedException(); } + public bool ResponseValidation { get => throw new NotImplementedException(); init => throw new NotImplementedException(); } + public int? MaxRetries { get => throw new NotImplementedException(); init => throw new NotImplementedException(); } + public TimeSpan? Timeout { get => throw new NotImplementedException(); init => throw new NotImplementedException(); } + public string? APIKey { get => throw new NotImplementedException(); init => throw new NotImplementedException(); } + public string? AuthToken { get => throw new NotImplementedException(); init => throw new NotImplementedException(); } + + public IMessageService Messages => throw new NotImplementedException(); + + public IModelService Models => throw new NotImplementedException(); + + public IBetaService Beta => this.BetaService; + + public IBetaService BetaService { get; } + + IMessageService IAnthropicClient.Messages => new Mock().Object; + + public Task Execute(HttpRequest request, CancellationToken cancellationToken = default) where T : ParamsBase + { + throw new NotImplementedException(); + } + + public IAnthropicClient WithOptions(Func modifier) + { + throw new NotImplementedException(); + } + + private sealed class TestBetaService : IBetaService + { + private readonly IAnthropicClient _client; + + public TestBetaService(IAnthropicClient client) + { + this._client = client; + } + + public global::Anthropic.Services.Beta.IModelService Models => throw new NotImplementedException(); + + public global::Anthropic.Services.Beta.IFileService Files => throw new NotImplementedException(); + + public global::Anthropic.Services.Beta.ISkillService Skills => throw new NotImplementedException(); + + public IBetaMessageService Messages => new Mock().Object; + + public IBetaService WithOptions(Func modifier) + { + throw new NotImplementedException(); + } + } + } +} diff --git a/dotnet/tests/Microsoft.Agents.AI.Anthropic.UnitTests/Extensions/AnthropicClientExtensionsTests.cs b/dotnet/tests/Microsoft.Agents.AI.Anthropic.UnitTests/Extensions/AnthropicClientExtensionsTests.cs new file mode 100644 index 0000000000..c8bf4d6a5e --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.Anthropic.UnitTests/Extensions/AnthropicClientExtensionsTests.cs @@ -0,0 +1,257 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Collections.Generic; +using System.Net.Http; +using System.Runtime.CompilerServices; +using System.Threading; +using System.Threading.Tasks; +using Anthropic; +using Anthropic.Core; +using Anthropic.Services; +using Microsoft.Extensions.AI; + +namespace Microsoft.Agents.AI.Anthropic.UnitTests.Extensions; + +/// +/// Unit tests for the AnthropicClientExtensions class. +/// +public sealed class AnthropicClientExtensionsTests +{ + /// + /// Test custom chat client that can be used to verify clientFactory functionality. + /// + private sealed class TestChatClient : IChatClient + { + private readonly IChatClient _innerClient; + + public TestChatClient(IChatClient innerClient) + { + this._innerClient = innerClient; + } + + public Task GetResponseAsync(IEnumerable messages, ChatOptions? options = null, CancellationToken cancellationToken = default) + => this._innerClient.GetResponseAsync(messages, options, cancellationToken); + + public async IAsyncEnumerable GetStreamingResponseAsync( + IEnumerable messages, ChatOptions? options = null, [EnumeratorCancellation] CancellationToken cancellationToken = default) + { + await foreach (var update in this._innerClient.GetStreamingResponseAsync(messages, options, cancellationToken)) + { + yield return update; + } + } + + public object? GetService(Type serviceType, object? serviceKey = null) + { + // Return this instance when requested + if (serviceType == typeof(TestChatClient)) + { + return this; + } + + return this._innerClient.GetService(serviceType, serviceKey); + } + + public void Dispose() => this._innerClient.Dispose(); + } + + /// + /// Creates a test ChatClient implementation for testing. + /// + private sealed class TestAnthropicChatClient : IAnthropicClient + { + public TestAnthropicChatClient() + { + } + + public HttpClient HttpClient { get => throw new NotImplementedException(); init => throw new NotImplementedException(); } + public Uri BaseUrl { get => new("http://localhost"); init => throw new NotImplementedException(); } + public bool ResponseValidation { get => throw new NotImplementedException(); init => throw new NotImplementedException(); } + public int? MaxRetries { get => throw new NotImplementedException(); init => throw new NotImplementedException(); } + public TimeSpan? Timeout { get => throw new NotImplementedException(); init => throw new NotImplementedException(); } + public string? APIKey { get => throw new NotImplementedException(); init => throw new NotImplementedException(); } + public string? AuthToken { get => throw new NotImplementedException(); init => throw new NotImplementedException(); } + + public IMessageService Messages => throw new NotImplementedException(); + + public IModelService Models => throw new NotImplementedException(); + + public IBetaService Beta => throw new NotImplementedException(); + + public Task Execute(HttpRequest request, CancellationToken cancellationToken = default) where T : ParamsBase + { + throw new NotImplementedException(); + } + + public IAnthropicClient WithOptions(Func modifier) + { + throw new NotImplementedException(); + } + } + + /// + /// Verify that CreateAIAgent with clientFactory parameter correctly applies the factory. + /// + [Fact] + public void CreateAIAgent_WithClientFactory_AppliesFactoryCorrectly() + { + // Arrange + var chatClient = new TestAnthropicChatClient(); + var testChatClient = new TestChatClient(chatClient.AsIChatClient()); + + // Act + var agent = chatClient.CreateAIAgent( + model: "test-model", + instructions: "Test instructions", + name: "Test Agent", + description: "Test description", + clientFactory: (innerClient) => testChatClient); + + // Assert + Assert.NotNull(agent); + Assert.Equal("Test Agent", agent.Name); + Assert.Equal("Test description", agent.Description); + + // Verify that the custom chat client can be retrieved from the agent's service collection + var retrievedTestClient = agent.GetService(); + Assert.NotNull(retrievedTestClient); + Assert.Same(testChatClient, retrievedTestClient); + } + + /// + /// Verify that CreateAIAgent with clientFactory using AsBuilder pattern works correctly. + /// + [Fact] + public void CreateAIAgent_WithClientFactoryUsingAsBuilder_AppliesFactoryCorrectly() + { + // Arrange + var chatClient = new TestAnthropicChatClient(); + TestChatClient? testChatClient = null; + + // Act + var agent = chatClient.CreateAIAgent( + model: "test-model", + instructions: "Test instructions", + clientFactory: (innerClient) => + innerClient.AsBuilder().Use((innerClient) => testChatClient = new TestChatClient(innerClient)).Build()); + + // Assert + Assert.NotNull(agent); + + // Verify that the custom chat client can be retrieved from the agent's service collection + var retrievedTestClient = agent.GetService(); + Assert.NotNull(retrievedTestClient); + Assert.Same(testChatClient, retrievedTestClient); + } + + /// + /// Verify that CreateAIAgent with options and clientFactory parameter correctly applies the factory. + /// + [Fact] + public void CreateAIAgent_WithOptionsAndClientFactory_AppliesFactoryCorrectly() + { + // Arrange + var chatClient = new TestAnthropicChatClient(); + var testChatClient = new TestChatClient(chatClient.AsIChatClient()); + var options = new ChatClientAgentOptions + { + Name = "Test Agent", + Description = "Test description", + ChatOptions = new() { Instructions = "Test instructions" } + }; + + // Act + var agent = chatClient.CreateAIAgent( + options, + clientFactory: (innerClient) => testChatClient); + + // Assert + Assert.NotNull(agent); + Assert.Equal("Test Agent", agent.Name); + Assert.Equal("Test description", agent.Description); + + // Verify that the custom chat client can be retrieved from the agent's service collection + var retrievedTestClient = agent.GetService(); + Assert.NotNull(retrievedTestClient); + Assert.Same(testChatClient, retrievedTestClient); + } + + /// + /// Verify that CreateAIAgent without clientFactory works normally. + /// + [Fact] + public void CreateAIAgent_WithoutClientFactory_WorksNormally() + { + // Arrange + var chatClient = new TestAnthropicChatClient(); + + // Act + var agent = chatClient.CreateAIAgent( + model: "test-model", + instructions: "Test instructions", + name: "Test Agent"); + + // Assert + Assert.NotNull(agent); + Assert.Equal("Test Agent", agent.Name); + + // Verify that no TestChatClient is available since no factory was provided + var retrievedTestClient = agent.GetService(); + Assert.Null(retrievedTestClient); + } + + /// + /// Verify that CreateAIAgent with null clientFactory works normally. + /// + [Fact] + public void CreateAIAgent_WithNullClientFactory_WorksNormally() + { + // Arrange + var chatClient = new TestAnthropicChatClient(); + + // Act + var agent = chatClient.CreateAIAgent( + model: "test-model", + instructions: "Test instructions", + name: "Test Agent", + clientFactory: null); + + // Assert + Assert.NotNull(agent); + Assert.Equal("Test Agent", agent.Name); + + // Verify that no TestChatClient is available since no factory was provided + var retrievedTestClient = agent.GetService(); + Assert.Null(retrievedTestClient); + } + + /// + /// Verify that CreateAIAgent throws ArgumentNullException when client is null. + /// + [Fact] + public void CreateAIAgent_WithNullClient_ThrowsArgumentNullException() + { + // Act & Assert + var exception = Assert.Throws(() => + ((TestAnthropicChatClient)null!).CreateAIAgent("test-model")); + + Assert.Equal("client", exception.ParamName); + } + + /// + /// Verify that CreateAIAgent with options throws ArgumentNullException when options is null. + /// + [Fact] + public void CreateAIAgent_WithNullOptions_ThrowsArgumentNullException() + { + // Arrange + var chatClient = new TestAnthropicChatClient(); + + // Act & Assert + var exception = Assert.Throws(() => + chatClient.CreateAIAgent((ChatClientAgentOptions)null!)); + + Assert.Equal("options", exception.ParamName); + } +} diff --git a/dotnet/tests/Microsoft.Agents.AI.Anthropic.UnitTests/Microsoft.Agents.AI.Anthropic.UnitTests.csproj b/dotnet/tests/Microsoft.Agents.AI.Anthropic.UnitTests/Microsoft.Agents.AI.Anthropic.UnitTests.csproj new file mode 100644 index 0000000000..291c56f879 --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.Anthropic.UnitTests/Microsoft.Agents.AI.Anthropic.UnitTests.csproj @@ -0,0 +1,11 @@ + + + + true + + + + + + + diff --git a/dotnet/tests/Microsoft.Agents.AI.AzureAI.Persistent.UnitTests/Extensions/PersistentAgentsClientExtensionsTests.cs b/dotnet/tests/Microsoft.Agents.AI.AzureAI.Persistent.UnitTests/Extensions/PersistentAgentsClientExtensionsTests.cs index 56b89d2df8..b661a392be 100644 --- a/dotnet/tests/Microsoft.Agents.AI.AzureAI.Persistent.UnitTests/Extensions/PersistentAgentsClientExtensionsTests.cs +++ b/dotnet/tests/Microsoft.Agents.AI.AzureAI.Persistent.UnitTests/Extensions/PersistentAgentsClientExtensionsTests.cs @@ -5,6 +5,7 @@ using System.Collections.Generic; using System.IO; using System.Linq; +using System.Reflection; using System.Threading; using System.Threading.Tasks; using Azure; @@ -309,7 +310,7 @@ public void GetAIAgent_WithResponseAndOptions_WorksCorrectly() { Name = "Override Name", Description = "Override Description", - Instructions = "Override Instructions" + ChatOptions = new() { Instructions = "Override Instructions" } }; // Act @@ -336,7 +337,7 @@ public void GetAIAgent_WithPersistentAgentAndOptions_WorksCorrectly() { Name = "Override Name", Description = "Override Description", - Instructions = "Override Instructions" + ChatOptions = new() { Instructions = "Override Instructions" } }; // Act @@ -385,7 +386,7 @@ public void GetAIAgent_WithAgentIdAndOptions_WorksCorrectly() { Name = "Override Name", Description = "Override Description", - Instructions = "Override Instructions" + ChatOptions = new() { Instructions = "Override Instructions" } }; // Act @@ -412,7 +413,7 @@ public async Task GetAIAgentAsync_WithAgentIdAndOptions_WorksCorrectlyAsync() { Name = "Override Name", Description = "Override Description", - Instructions = "Override Instructions" + ChatOptions = new() { Instructions = "Override Instructions" } }; // Act @@ -556,7 +557,7 @@ public void CreateAIAgent_WithOptions_WorksCorrectly() { Name = "Test Agent", Description = "Test description", - Instructions = "Test instructions" + ChatOptions = new() { Instructions = "Test instructions" } }; // Act @@ -583,7 +584,7 @@ public async Task CreateAIAgentAsync_WithOptions_WorksCorrectlyAsync() { Name = "Test Agent", Description = "Test description", - Instructions = "Test instructions" + ChatOptions = new() { Instructions = "Test instructions" } }; // Act @@ -726,6 +727,159 @@ public async Task CreateAIAgentAsync_WithEmptyModel_ThrowsArgumentExceptionAsync Assert.Equal("model", exception.ParamName); } + /// + /// Verify that CreateAIAgent with services parameter correctly passes it through to the ChatClientAgent. + /// + [Fact] + public void CreateAIAgent_WithServices_PassesServicesToAgent() + { + // Arrange + var client = CreateFakePersistentAgentsClient(); + var serviceProvider = new TestServiceProvider(); + const string Model = "test-model"; + + // Act + var agent = client.CreateAIAgent( + Model, + instructions: "Test instructions", + name: "Test Agent", + services: serviceProvider); + + // Assert + Assert.NotNull(agent); + + // Verify the IServiceProvider was passed through to the FunctionInvokingChatClient + var chatClient = agent.GetService(); + Assert.NotNull(chatClient); + var functionInvokingClient = chatClient.GetService(); + Assert.NotNull(functionInvokingClient); + Assert.Same(serviceProvider, GetFunctionInvocationServices(functionInvokingClient)); + } + + /// + /// Verify that CreateAIAgentAsync with services parameter correctly passes it through to the ChatClientAgent. + /// + [Fact] + public async Task CreateAIAgentAsync_WithServices_PassesServicesToAgentAsync() + { + // Arrange + var client = CreateFakePersistentAgentsClient(); + var serviceProvider = new TestServiceProvider(); + const string Model = "test-model"; + + // Act + var agent = await client.CreateAIAgentAsync( + Model, + instructions: "Test instructions", + name: "Test Agent", + services: serviceProvider); + + // Assert + Assert.NotNull(agent); + + // Verify the IServiceProvider was passed through to the FunctionInvokingChatClient + var chatClient = agent.GetService(); + Assert.NotNull(chatClient); + var functionInvokingClient = chatClient.GetService(); + Assert.NotNull(functionInvokingClient); + Assert.Same(serviceProvider, GetFunctionInvocationServices(functionInvokingClient)); + } + + /// + /// Verify that GetAIAgent with services parameter correctly passes it through to the ChatClientAgent. + /// + [Fact] + public void GetAIAgent_WithServices_PassesServicesToAgent() + { + // Arrange + var client = CreateFakePersistentAgentsClient(); + var serviceProvider = new TestServiceProvider(); + + // Act + var agent = client.GetAIAgent("agent_abc123", services: serviceProvider); + + // Assert + Assert.NotNull(agent); + + // Verify the IServiceProvider was passed through to the FunctionInvokingChatClient + var chatClient = agent.GetService(); + Assert.NotNull(chatClient); + var functionInvokingClient = chatClient.GetService(); + Assert.NotNull(functionInvokingClient); + Assert.Same(serviceProvider, GetFunctionInvocationServices(functionInvokingClient)); + } + + /// + /// Verify that GetAIAgentAsync with services parameter correctly passes it through to the ChatClientAgent. + /// + [Fact] + public async Task GetAIAgentAsync_WithServices_PassesServicesToAgentAsync() + { + // Arrange + var client = CreateFakePersistentAgentsClient(); + var serviceProvider = new TestServiceProvider(); + + // Act + var agent = await client.GetAIAgentAsync("agent_abc123", services: serviceProvider); + + // Assert + Assert.NotNull(agent); + + // Verify the IServiceProvider was passed through to the FunctionInvokingChatClient + var chatClient = agent.GetService(); + Assert.NotNull(chatClient); + var functionInvokingClient = chatClient.GetService(); + Assert.NotNull(functionInvokingClient); + Assert.Same(serviceProvider, GetFunctionInvocationServices(functionInvokingClient)); + } + + /// + /// Verify that CreateAIAgent with both clientFactory and services works correctly. + /// + [Fact] + public void CreateAIAgent_WithClientFactoryAndServices_AppliesBothCorrectly() + { + // Arrange + var client = CreateFakePersistentAgentsClient(); + var serviceProvider = new TestServiceProvider(); + TestChatClient? testChatClient = null; + const string Model = "test-model"; + + // Act + var agent = client.CreateAIAgent( + Model, + instructions: "Test instructions", + name: "Test Agent", + clientFactory: (innerClient) => testChatClient = new TestChatClient(innerClient), + services: serviceProvider); + + // Assert + Assert.NotNull(agent); + + // Verify the custom chat client was applied + var retrievedTestClient = agent.GetService(); + Assert.NotNull(retrievedTestClient); + Assert.Same(testChatClient, retrievedTestClient); + + // Verify the IServiceProvider was passed through + var chatClient = agent.GetService(); + Assert.NotNull(chatClient); + var functionInvokingClient = chatClient.GetService(); + Assert.NotNull(functionInvokingClient); + Assert.Same(serviceProvider, GetFunctionInvocationServices(functionInvokingClient)); + } + + /// + /// Uses reflection to access the FunctionInvocationServices property which is not public. + /// + private static IServiceProvider? GetFunctionInvocationServices(FunctionInvokingChatClient client) + { + var property = typeof(FunctionInvokingChatClient).GetProperty( + "FunctionInvocationServices", + BindingFlags.Instance | BindingFlags.Public | BindingFlags.NonPublic); + return property?.GetValue(client) as IServiceProvider; + } + /// /// Test custom chat client that can be used to verify clientFactory functionality. /// @@ -736,6 +890,14 @@ public TestChatClient(IChatClient innerClient) : base(innerClient) } } + /// + /// A simple test IServiceProvider implementation for testing. + /// + private sealed class TestServiceProvider : IServiceProvider + { + public object? GetService(Type serviceType) => null; + } + public sealed class FakePersistentAgentsAdministrationClient : PersistentAgentsAdministrationClient { public FakePersistentAgentsAdministrationClient() @@ -761,7 +923,7 @@ private static PersistentAgentsClient CreateFakePersistentAgentsClient() { var client = new PersistentAgentsClient("https://any.com", DelegatedTokenCredential.Create((_, _) => new AccessToken())); - ((System.Reflection.TypeInfo)typeof(PersistentAgentsClient)).DeclaredFields.First(f => f.Name == "_client") + ((TypeInfo)typeof(PersistentAgentsClient)).DeclaredFields.First(f => f.Name == "_client") .SetValue(client, new FakePersistentAgentsAdministrationClient()); return client; } diff --git a/dotnet/tests/Microsoft.Agents.AI.AzureAI.Persistent.UnitTests/Microsoft.Agents.AI.AzureAI.Persistent.UnitTests.csproj b/dotnet/tests/Microsoft.Agents.AI.AzureAI.Persistent.UnitTests/Microsoft.Agents.AI.AzureAI.Persistent.UnitTests.csproj index 80c0086675..ca33d52d6b 100644 --- a/dotnet/tests/Microsoft.Agents.AI.AzureAI.Persistent.UnitTests/Microsoft.Agents.AI.AzureAI.Persistent.UnitTests.csproj +++ b/dotnet/tests/Microsoft.Agents.AI.AzureAI.Persistent.UnitTests/Microsoft.Agents.AI.AzureAI.Persistent.UnitTests.csproj @@ -1,9 +1,5 @@ - - $(ProjectsTargetFrameworks) - - diff --git a/dotnet/tests/Microsoft.Agents.AI.AzureAI.UnitTests/AzureAIProjectChatClientExtensionsTests.cs b/dotnet/tests/Microsoft.Agents.AI.AzureAI.UnitTests/AzureAIProjectChatClientExtensionsTests.cs new file mode 100644 index 0000000000..4ca2b7f461 --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.AzureAI.UnitTests/AzureAIProjectChatClientExtensionsTests.cs @@ -0,0 +1,2819 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.ClientModel; +using System.ClientModel.Primitives; +using System.Collections.Generic; +using System.IO; +using System.Linq; +using System.Net; +using System.Net.Http; +using System.Text; +using System.Text.Json; +using System.Threading; +using System.Threading.Tasks; +using Azure.AI.Projects; +using Azure.AI.Projects.OpenAI; +using Microsoft.Extensions.AI; +using Moq; +using OpenAI.Responses; + +namespace Microsoft.Agents.AI.AzureAI.UnitTests; + +/// +/// Unit tests for the class. +/// +public sealed class AzureAIProjectChatClientExtensionsTests +{ + #region GetAIAgent(AIProjectClient, AgentRecord) Tests + + /// + /// Verify that GetAIAgent throws ArgumentNullException when AIProjectClient is null. + /// + [Fact] + public void GetAIAgent_WithAgentRecord_WithNullClient_ThrowsArgumentNullException() + { + // Arrange + AIProjectClient? client = null; + AgentRecord agentRecord = this.CreateTestAgentRecord(); + + // Act & Assert + var exception = Assert.Throws(() => + client!.GetAIAgent(agentRecord)); + + Assert.Equal("aiProjectClient", exception.ParamName); + } + + /// + /// Verify that GetAIAgent throws ArgumentNullException when agentRecord is null. + /// + [Fact] + public void GetAIAgent_WithAgentRecord_WithNullAgentRecord_ThrowsArgumentNullException() + { + // Arrange + var mockClient = new Mock(); + + // Act & Assert + var exception = Assert.Throws(() => + mockClient.Object.GetAIAgent((AgentRecord)null!)); + + Assert.Equal("agentRecord", exception.ParamName); + } + + /// + /// Verify that GetAIAgent with AgentRecord creates a valid agent. + /// + [Fact] + public void GetAIAgent_WithAgentRecord_CreatesValidAgent() + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(); + AgentRecord agentRecord = this.CreateTestAgentRecord(); + + // Act + var agent = client.GetAIAgent(agentRecord); + + // Assert + Assert.NotNull(agent); + Assert.Equal("agent_abc123", agent.Name); + } + + /// + /// Verify that GetAIAgent with AgentRecord and clientFactory applies the factory. + /// + [Fact] + public void GetAIAgent_WithAgentRecord_WithClientFactory_AppliesFactoryCorrectly() + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(); + AgentRecord agentRecord = this.CreateTestAgentRecord(); + TestChatClient? testChatClient = null; + + // Act + var agent = client.GetAIAgent( + agentRecord, + clientFactory: (innerClient) => testChatClient = new TestChatClient(innerClient)); + + // Assert + Assert.NotNull(agent); + var retrievedTestClient = agent.GetService(); + Assert.NotNull(retrievedTestClient); + Assert.Same(testChatClient, retrievedTestClient); + } + + #endregion + + #region GetAIAgent(AIProjectClient, AgentVersion) Tests + + /// + /// Verify that GetAIAgent throws ArgumentNullException when AIProjectClient is null. + /// + [Fact] + public void GetAIAgent_WithAgentVersion_WithNullClient_ThrowsArgumentNullException() + { + // Arrange + AIProjectClient? client = null; + AgentVersion agentVersion = this.CreateTestAgentVersion(); + + // Act & Assert + var exception = Assert.Throws(() => + client!.GetAIAgent(agentVersion)); + + Assert.Equal("aiProjectClient", exception.ParamName); + } + + /// + /// Verify that GetAIAgent throws ArgumentNullException when agentVersion is null. + /// + [Fact] + public void GetAIAgent_WithAgentVersion_WithNullAgentVersion_ThrowsArgumentNullException() + { + // Arrange + var mockClient = new Mock(); + + // Act & Assert + var exception = Assert.Throws(() => + mockClient.Object.GetAIAgent((AgentVersion)null!)); + + Assert.Equal("agentVersion", exception.ParamName); + } + + /// + /// Verify that GetAIAgent with AgentVersion creates a valid agent. + /// + [Fact] + public void GetAIAgent_WithAgentVersion_CreatesValidAgent() + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(); + AgentVersion agentVersion = this.CreateTestAgentVersion(); + + // Act + var agent = client.GetAIAgent(agentVersion); + + // Assert + Assert.NotNull(agent); + Assert.Equal("agent_abc123", agent.Name); + } + + /// + /// Verify that GetAIAgent with AgentVersion and clientFactory applies the factory. + /// + [Fact] + public void GetAIAgent_WithAgentVersion_WithClientFactory_AppliesFactoryCorrectly() + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(); + AgentVersion agentVersion = this.CreateTestAgentVersion(); + TestChatClient? testChatClient = null; + + // Act + var agent = client.GetAIAgent( + agentVersion, + clientFactory: (innerClient) => testChatClient = new TestChatClient(innerClient)); + + // Assert + Assert.NotNull(agent); + var retrievedTestClient = agent.GetService(); + Assert.NotNull(retrievedTestClient); + Assert.Same(testChatClient, retrievedTestClient); + } + + /// + /// Verify that GetAIAgent with requireInvocableTools=true enforces invocable tools. + /// + [Fact] + public void GetAIAgent_WithAgentVersion_WithRequireInvocableToolsTrue_EnforcesInvocableTools() + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(); + AgentVersion agentVersion = this.CreateTestAgentVersion(); + var tools = new List + { + AIFunctionFactory.Create(() => "test", "test_function", "A test function") + }; + + // Act + var agent = client.GetAIAgent(agentVersion, tools: tools); + + // Assert + Assert.NotNull(agent); + Assert.IsType(agent); + } + + /// + /// Verify that GetAIAgent with requireInvocableTools=false allows declarative functions. + /// + [Fact] + public void GetAIAgent_WithAgentVersion_WithRequireInvocableToolsFalse_AllowsDeclarativeFunctions() + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(); + AgentVersion agentVersion = this.CreateTestAgentVersion(); + + // Act - should not throw even without tools when requireInvocableTools is false + var agent = client.GetAIAgent(agentVersion); + + // Assert + Assert.NotNull(agent); + Assert.IsType(agent); + } + + #endregion + + #region GetAIAgent(AIProjectClient, ChatClientAgentOptions) Tests + + /// + /// Verify that GetAIAgent with ChatClientAgentOptions throws ArgumentNullException when client is null. + /// + [Fact] + public void GetAIAgent_WithOptions_WithNullClient_ThrowsArgumentNullException() + { + // Arrange + AIProjectClient? client = null; + var options = new ChatClientAgentOptions { Name = "test-agent" }; + + // Act & Assert + var exception = Assert.Throws(() => + client!.GetAIAgent(options)); + + Assert.Equal("aiProjectClient", exception.ParamName); + } + + /// + /// Verify that GetAIAgent with ChatClientAgentOptions throws ArgumentNullException when options is null. + /// + [Fact] + public void GetAIAgent_WithOptions_WithNullOptions_ThrowsArgumentNullException() + { + // Arrange + var mockClient = new Mock(); + + // Act & Assert + var exception = Assert.Throws(() => + mockClient.Object.GetAIAgent((ChatClientAgentOptions)null!)); + + Assert.Equal("options", exception.ParamName); + } + + /// + /// Verify that GetAIAgent with ChatClientAgentOptions throws ArgumentException when options.Name is null. + /// + [Fact] + public void GetAIAgent_WithOptions_WithoutName_ThrowsArgumentException() + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(); + var options = new ChatClientAgentOptions(); + + // Act & Assert + var exception = Assert.Throws(() => + client.GetAIAgent(options)); + + Assert.Contains("Agent name must be provided", exception.Message); + } + + /// + /// Verify that GetAIAgent with ChatClientAgentOptions creates a valid agent. + /// + [Fact] + public void GetAIAgent_WithOptions_CreatesValidAgent() + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(agentName: "test-agent"); + var options = new ChatClientAgentOptions { Name = "test-agent" }; + + // Act + var agent = client.GetAIAgent(options); + + // Assert + Assert.NotNull(agent); + Assert.Equal("test-agent", agent.Name); + } + + /// + /// Verify that GetAIAgent with ChatClientAgentOptions and clientFactory applies the factory. + /// + [Fact] + public void GetAIAgent_WithOptions_WithClientFactory_AppliesFactoryCorrectly() + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(agentName: "test-agent"); + var options = new ChatClientAgentOptions { Name = "test-agent" }; + TestChatClient? testChatClient = null; + + // Act + var agent = client.GetAIAgent( + options, + clientFactory: (innerClient) => testChatClient = new TestChatClient(innerClient)); + + // Assert + Assert.NotNull(agent); + var retrievedTestClient = agent.GetService(); + Assert.NotNull(retrievedTestClient); + Assert.Same(testChatClient, retrievedTestClient); + } + + #endregion + + #region GetAIAgentAsync(AIProjectClient, ChatClientAgentOptions) Tests + + /// + /// Verify that GetAIAgentAsync with ChatClientAgentOptions throws ArgumentNullException when client is null. + /// + [Fact] + public async Task GetAIAgentAsync_WithOptions_WithNullClient_ThrowsArgumentNullExceptionAsync() + { + // Arrange + AIProjectClient? client = null; + var options = new ChatClientAgentOptions { Name = "test-agent" }; + + // Act & Assert + var exception = await Assert.ThrowsAsync(() => + client!.GetAIAgentAsync(options)); + + Assert.Equal("aiProjectClient", exception.ParamName); + } + + /// + /// Verify that GetAIAgentAsync with ChatClientAgentOptions throws ArgumentNullException when options is null. + /// + [Fact] + public async Task GetAIAgentAsync_WithOptions_WithNullOptions_ThrowsArgumentNullExceptionAsync() + { + // Arrange + var mockClient = new Mock(); + + // Act & Assert + var exception = await Assert.ThrowsAsync(() => + mockClient.Object.GetAIAgentAsync((ChatClientAgentOptions)null!)); + + Assert.Equal("options", exception.ParamName); + } + + /// + /// Verify that GetAIAgentAsync with ChatClientAgentOptions creates a valid agent. + /// + [Fact] + public async Task GetAIAgentAsync_WithOptions_CreatesValidAgentAsync() + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(agentName: "test-agent"); + var options = new ChatClientAgentOptions { Name = "test-agent" }; + + // Act + var agent = await client.GetAIAgentAsync(options); + + // Assert + Assert.NotNull(agent); + Assert.Equal("test-agent", agent.Name); + } + + #endregion + + #region GetAIAgent(AIProjectClient, string) Tests + + /// + /// Verify that GetAIAgent throws ArgumentNullException when AIProjectClient is null. + /// + [Fact] + public void GetAIAgent_ByName_WithNullClient_ThrowsArgumentNullException() + { + // Arrange + AIProjectClient? client = null; + + // Act & Assert + var exception = Assert.Throws(() => + client!.GetAIAgent("test-agent")); + + Assert.Equal("aiProjectClient", exception.ParamName); + } + + /// + /// Verify that GetAIAgent throws ArgumentNullException when name is null. + /// + [Fact] + public void GetAIAgent_ByName_WithNullName_ThrowsArgumentNullException() + { + // Arrange + var mockClient = new Mock(); + + // Act & Assert + var exception = Assert.Throws(() => + mockClient.Object.GetAIAgent((string)null!)); + + Assert.Equal("name", exception.ParamName); + } + + /// + /// Verify that GetAIAgent throws ArgumentException when name is empty. + /// + [Fact] + public void GetAIAgent_ByName_WithEmptyName_ThrowsArgumentException() + { + // Arrange + var mockClient = new Mock(); + + // Act & Assert + var exception = Assert.Throws(() => + mockClient.Object.GetAIAgent(string.Empty)); + + Assert.Equal("name", exception.ParamName); + } + + /// + /// Verify that GetAIAgent throws InvalidOperationException when agent is not found. + /// + [Fact] + public void GetAIAgent_ByName_WithNonExistentAgent_ThrowsInvalidOperationException() + { + // Arrange + var mockAgentOperations = new Mock(); + mockAgentOperations + .Setup(c => c.GetAgent(It.IsAny(), It.IsAny())) + .Returns(ClientResult.FromOptionalValue((AgentRecord)null!, new MockPipelineResponse(200, BinaryData.FromString("null")))); + + var mockClient = new Mock(); + mockClient.SetupGet(x => x.Agents).Returns(mockAgentOperations.Object); + mockClient.Setup(x => x.GetConnection(It.IsAny())).Returns(new ClientConnection("fake-connection-id", "http://localhost", ClientPipeline.Create(), CredentialKind.None)); + + // Act & Assert + var exception = Assert.Throws(() => + mockClient.Object.GetAIAgent("non-existent-agent")); + + Assert.Contains("not found", exception.Message); + } + + #endregion + + #region GetAIAgentAsync(AIProjectClient, string) Tests + + /// + /// Verify that GetAIAgentAsync throws ArgumentNullException when AIProjectClient is null. + /// + [Fact] + public async Task GetAIAgentAsync_ByName_WithNullClient_ThrowsArgumentNullExceptionAsync() + { + // Arrange + AIProjectClient? client = null; + + // Act & Assert + var exception = await Assert.ThrowsAsync(() => + client!.GetAIAgentAsync("test-agent")); + + Assert.Equal("aiProjectClient", exception.ParamName); + } + + /// + /// Verify that GetAIAgentAsync throws ArgumentNullException when name is null. + /// + [Fact] + public async Task GetAIAgentAsync_ByName_WithNullName_ThrowsArgumentNullExceptionAsync() + { + // Arrange + var mockClient = new Mock(); + + // Act & Assert + var exception = await Assert.ThrowsAsync(() => + mockClient.Object.GetAIAgentAsync(name: null!)); + + Assert.Equal("name", exception.ParamName); + } + + /// + /// Verify that GetAIAgentAsync throws InvalidOperationException when agent is not found. + /// + [Fact] + public async Task GetAIAgentAsync_ByName_WithNonExistentAgent_ThrowsInvalidOperationExceptionAsync() + { + // Arrange + var mockAgentOperations = new Mock(); + mockAgentOperations + .Setup(c => c.GetAgentAsync(It.IsAny(), It.IsAny())) + .ReturnsAsync(ClientResult.FromOptionalValue((AgentRecord)null!, new MockPipelineResponse(200, BinaryData.FromString("null")))); + + var mockClient = new Mock(); + mockClient.SetupGet(c => c.Agents).Returns(mockAgentOperations.Object); + mockClient.Setup(x => x.GetConnection(It.IsAny())).Returns(new ClientConnection("fake-connection-id", "http://localhost", ClientPipeline.Create(), CredentialKind.None)); + + // Act & Assert + var exception = await Assert.ThrowsAsync(() => + mockClient.Object.GetAIAgentAsync("non-existent-agent")); + + Assert.Contains("not found", exception.Message); + } + + #endregion + + #region GetAIAgent(AIProjectClient, AgentRecord) with tools Tests + + /// + /// Verify that GetAIAgent with additional tools when the definition has no tools does not throw and results in an agent with no tools. + /// + [Fact] + public void GetAIAgent_WithAgentRecordAndAdditionalTools_WhenDefinitionHasNoTools_ShouldNotThrow() + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(); + AgentRecord agentRecord = this.CreateTestAgentRecord(); + var tools = new List + { + AIFunctionFactory.Create(() => "test", "test_function", "A test function") + }; + + // Act + var agent = client.GetAIAgent(agentRecord, tools: tools); + + // Assert + Assert.NotNull(agent); + Assert.IsType(agent); + var chatClient = agent.GetService(); + Assert.NotNull(chatClient); + var agentVersion = chatClient.GetService(); + Assert.NotNull(agentVersion); + var definition = Assert.IsType(agentVersion.Definition); + Assert.Empty(definition.Tools); + } + + /// + /// Verify that GetAIAgent with null tools works correctly. + /// + [Fact] + public void GetAIAgent_WithAgentRecordAndNullTools_WorksCorrectly() + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(); + AgentRecord agentRecord = this.CreateTestAgentRecord(); + + // Act + var agent = client.GetAIAgent(agentRecord, tools: null); + + // Assert + Assert.NotNull(agent); + Assert.Equal("agent_abc123", agent.Name); + } + + #endregion + + #region GetAIAgentAsync(AIProjectClient, string) with tools Tests + + /// + /// Verify that GetAIAgentAsync with tools parameter creates an agent. + /// + [Fact] + public async Task GetAIAgentAsync_WithNameAndTools_CreatesAgentAsync() + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(); + var tools = new List + { + AIFunctionFactory.Create(() => "test", "test_function", "A test function") + }; + + // Act + var agent = await client.GetAIAgentAsync("test-agent", tools: tools); + + // Assert + Assert.NotNull(agent); + Assert.IsType(agent); + } + + #endregion + + #region CreateAIAgent(AIProjectClient, string, string) Tests + + /// + /// Verify that CreateAIAgent throws ArgumentNullException when AIProjectClient is null. + /// + [Fact] + public void CreateAIAgent_WithBasicParams_WithNullClient_ThrowsArgumentNullException() + { + // Arrange + AIProjectClient? client = null; + + // Act & Assert + var exception = Assert.Throws(() => + client!.CreateAIAgent("test-agent", "model", "instructions")); + + Assert.Equal("aiProjectClient", exception.ParamName); + } + + /// + /// Verify that CreateAIAgent throws ArgumentNullException when name is null. + /// + [Fact] + public void CreateAIAgent_WithBasicParams_WithNullName_ThrowsArgumentNullException() + { + // Arrange + var mockClient = new Mock(); + + // Act & Assert + var exception = Assert.Throws(() => + mockClient.Object.CreateAIAgent(null!, "model", "instructions")); + + Assert.Equal("name", exception.ParamName); + } + + #endregion + + #region CreateAIAgent(AIProjectClient, string, AgentDefinition) Tests + + /// + /// Verify that CreateAIAgent throws ArgumentNullException when AIProjectClient is null. + /// + [Fact] + public void CreateAIAgent_WithAgentDefinition_WithNullClient_ThrowsArgumentNullException() + { + // Arrange + AIProjectClient? client = null; + var definition = new PromptAgentDefinition("test-model"); + var options = new AgentVersionCreationOptions(definition); + + // Act & Assert + var exception = Assert.Throws(() => + client!.CreateAIAgent("test-agent", options)); + + Assert.Equal("aiProjectClient", exception.ParamName); + } + + /// + /// Verify that CreateAIAgent throws ArgumentNullException when name is null. + /// + [Fact] + public void CreateAIAgent_WithAgentDefinition_WithNullName_ThrowsArgumentNullException() + { + // Arrange + var mockClient = new Mock(); + var definition = new PromptAgentDefinition("test-model"); + var options = new AgentVersionCreationOptions(definition); + + // Act & Assert + var exception = Assert.Throws(() => + mockClient.Object.CreateAIAgent(null!, options)); + + Assert.Equal("name", exception.ParamName); + } + + /// + /// Verify that CreateAIAgent throws ArgumentNullException when creationOptions is null. + /// + [Fact] + public void CreateAIAgent_WithAgentDefinition_WithNullDefinition_ThrowsArgumentNullException() + { + // Arrange + var mockClient = new Mock(); + + // Act & Assert + var exception = Assert.Throws(() => + mockClient.Object.CreateAIAgent("test-agent", (AgentVersionCreationOptions)null!)); + + Assert.Equal("creationOptions", exception.ParamName); + } + + #endregion + + #region CreateAIAgent(AIProjectClient, ChatClientAgentOptions, string) Tests + + /// + /// Verify that CreateAIAgent throws ArgumentNullException when AIProjectClient is null. + /// + [Fact] + public void CreateAIAgent_WithOptions_WithNullClient_ThrowsArgumentNullException() + { + // Arrange + AIProjectClient? client = null; + var options = new ChatClientAgentOptions { Name = "test-agent" }; + + // Act & Assert + var exception = Assert.Throws(() => + client!.CreateAIAgent("model", options)); + + Assert.Equal("aiProjectClient", exception.ParamName); + } + + /// + /// Verify that CreateAIAgent throws ArgumentNullException when options is null. + /// + [Fact] + public void CreateAIAgent_WithOptions_WithNullOptions_ThrowsArgumentNullException() + { + // Arrange + var mockClient = new Mock(); + + // Act & Assert + var exception = Assert.Throws(() => + mockClient.Object.CreateAIAgent("model", (ChatClientAgentOptions)null!)); + + Assert.Equal("options", exception.ParamName); + } + + /// + /// Verify that CreateAIAgent throws ArgumentNullException when model is null. + /// + [Fact] + public void CreateAIAgent_WithOptions_WithNullModel_ThrowsArgumentNullException() + { + // Arrange + var mockClient = new Mock(); + var options = new ChatClientAgentOptions { Name = "test-agent" }; + + // Act & Assert + var exception = Assert.Throws(() => + mockClient.Object.CreateAIAgent(null!, options)); + + Assert.Equal("model", exception.ParamName); + } + + /// + /// Verify that CreateAIAgent throws ArgumentNullException when options.Name is null. + /// + [Fact] + public void CreateAIAgent_WithOptions_WithoutName_ThrowsException() + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(); + var options = new ChatClientAgentOptions(); + + // Act & Assert + var exception = Assert.Throws(() => + client.CreateAIAgent("test-model", options)); + + Assert.Contains("Agent name must be provided", exception.Message); + } + + /// + /// Verify that CreateAIAgent with model and options creates a valid agent. + /// + [Fact] + public void CreateAIAgent_WithModelAndOptions_CreatesValidAgent() + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(agentName: "test-agent", instructions: "Test instructions"); + var options = new ChatClientAgentOptions + { + Name = "test-agent", + ChatOptions = new() { Instructions = "Test instructions" } + }; + + // Act + var agent = client.CreateAIAgent("test-model", options); + + // Assert + Assert.NotNull(agent); + Assert.Equal("test-agent", agent.Name); + Assert.Equal("Test instructions", agent.Instructions); + } + + /// + /// Verify that CreateAIAgent with model and options and clientFactory applies the factory. + /// + [Fact] + public void CreateAIAgent_WithModelAndOptions_WithClientFactory_AppliesFactoryCorrectly() + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(agentName: "test-agent", instructions: "Test instructions"); + var options = new ChatClientAgentOptions + { + Name = "test-agent", + ChatOptions = new() { Instructions = "Test instructions" } + }; + TestChatClient? testChatClient = null; + + // Act + var agent = client.CreateAIAgent( + "test-model", + options, + clientFactory: (innerClient) => testChatClient = new TestChatClient(innerClient)); + + // Assert + Assert.NotNull(agent); + var retrievedTestClient = agent.GetService(); + Assert.NotNull(retrievedTestClient); + Assert.Same(testChatClient, retrievedTestClient); + } + + /// + /// Verify that CreateAIAgentAsync with model and options creates a valid agent. + /// + [Fact] + public async Task CreateAIAgentAsync_WithModelAndOptions_CreatesValidAgentAsync() + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(agentName: "test-agent", instructions: "Test instructions"); + var options = new ChatClientAgentOptions + { + Name = "test-agent", + ChatOptions = new() { Instructions = "Test instructions" } + }; + + // Act + var agent = await client.CreateAIAgentAsync("test-model", options); + + // Assert + Assert.NotNull(agent); + Assert.Equal("test-agent", agent.Name); + Assert.Equal("Test instructions", agent.Instructions); + } + + /// + /// Verify that CreateAIAgentAsync with model and options and clientFactory applies the factory. + /// + [Fact] + public async Task CreateAIAgentAsync_WithModelAndOptions_WithClientFactory_AppliesFactoryCorrectlyAsync() + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(agentName: "test-agent", instructions: "Test instructions"); + var options = new ChatClientAgentOptions + { + Name = "test-agent", + ChatOptions = new() { Instructions = "Test instructions" } + }; + TestChatClient? testChatClient = null; + + // Act + var agent = await client.CreateAIAgentAsync( + "test-model", + options, + clientFactory: (innerClient) => testChatClient = new TestChatClient(innerClient)); + + // Assert + Assert.NotNull(agent); + var retrievedTestClient = agent.GetService(); + Assert.NotNull(retrievedTestClient); + Assert.Same(testChatClient, retrievedTestClient); + } + + #endregion + + #region CreateAIAgentAsync(AIProjectClient, string, AgentDefinition) Tests + + /// + /// Verify that CreateAIAgentAsync throws ArgumentNullException when AIProjectClient is null. + /// + [Fact] + public async Task CreateAIAgentAsync_WithAgentDefinition_WithNullClient_ThrowsArgumentNullExceptionAsync() + { + // Arrange + AIProjectClient? client = null; + var definition = new PromptAgentDefinition("test-model"); + var options = new AgentVersionCreationOptions(definition); + + // Act & Assert + var exception = await Assert.ThrowsAsync(() => + client!.CreateAIAgentAsync("agent-name", options)); + + Assert.Equal("aiProjectClient", exception.ParamName); + } + + /// + /// Verify that CreateAIAgentAsync throws ArgumentNullException when creationOptions is null. + /// + [Fact] + public async Task CreateAIAgentAsync_WithAgentDefinition_WithNullDefinition_ThrowsArgumentNullExceptionAsync() + { + // Arrange + var mockClient = new Mock(); + + // Act & Assert + var exception = await Assert.ThrowsAsync(() => + mockClient.Object.CreateAIAgentAsync(name: "agent-name", null!)); + + Assert.Equal("creationOptions", exception.ParamName); + } + + #endregion + + #region Tool Validation Tests + + /// + /// Verify that CreateAIAgent creates an agent successfully. + /// + [Fact] + public void CreateAIAgent_WithDefinition_CreatesAgentSuccessfully() + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(); + var definition = new PromptAgentDefinition("test-model") { Instructions = "Test" }; + var options = new AgentVersionCreationOptions(definition); + + // Act + var agent = client.CreateAIAgent("test-agent", options); + + // Assert + Assert.NotNull(agent); + Assert.IsType(agent); + } + + /// + /// Verify that CreateAIAgent without tools parameter creates an agent successfully. + /// + [Fact] + public void CreateAIAgent_WithoutToolsParameter_CreatesAgentSuccessfully() + { + // Arrange + var definition = new PromptAgentDefinition("test-model") { Instructions = "Test" }; + + var definitionResponse = GeneratePromptDefinitionResponse(definition, null); + AIProjectClient client = this.CreateTestAgentClient(agentName: "test-agent", agentDefinitionResponse: definitionResponse); + + var options = new AgentVersionCreationOptions(definition); + + // Act + var agent = client.CreateAIAgent("test-agent", options); + + // Assert + Assert.NotNull(agent); + Assert.IsType(agent); + } + + /// + /// Verify that CreateAIAgent without tools in definition creates an agent successfully. + /// + [Fact] + public void CreateAIAgent_WithoutToolsInDefinition_CreatesAgentSuccessfully() + { + // Arrange + var definition = new PromptAgentDefinition("test-model") { Instructions = "Test" }; + AIProjectClient client = this.CreateTestAgentClient(agentName: "test-agent", agentDefinitionResponse: definition); + + var options = new AgentVersionCreationOptions(definition); + + // Act + var agent = client.CreateAIAgent("test-agent", options); + + // Assert + Assert.NotNull(agent); + Assert.IsType(agent); + } + + /// + /// Verify that CreateAIAgent uses tools from the definition when no separate tools parameter is provided. + /// + [Fact] + public void CreateAIAgent_WithDefinitionTools_UsesDefinitionTools() + { + // Arrange + var definition = new PromptAgentDefinition("test-model") { Instructions = "Test" }; + + // Add a function tool to the definition + definition.Tools.Add(ResponseTool.CreateFunctionTool("required_tool", BinaryData.FromString("{}"), strictModeEnabled: false)); + + // Create a response definition with the same tool + var definitionResponse = GeneratePromptDefinitionResponse(definition, definition.Tools.Select(t => t.AsAITool()).ToList()); + AIProjectClient client = this.CreateTestAgentClient(agentName: "test-agent", agentDefinitionResponse: definitionResponse); + + var options = new AgentVersionCreationOptions(definition); + + // Act + var agent = client.CreateAIAgent("test-agent", options); + + // Assert + Assert.NotNull(agent); + Assert.IsType(agent); + var agentVersion = agent.GetService(); + Assert.NotNull(agentVersion); + if (agentVersion.Definition is PromptAgentDefinition promptDef) + { + Assert.NotEmpty(promptDef.Tools); + Assert.Single(promptDef.Tools); + Assert.Equal("required_tool", (promptDef.Tools.First() as FunctionTool)?.FunctionName); + } + } + + /// + /// Verify that CreateAIAgentAsync when AI Tools are provided, uses them for the definition via http request. + /// + [Fact] + public async Task CreateAIAgentAsync_WithNameAndAITools_SendsToolDefinitionViaHttpAsync() + { + // Arrange + using var httpHandler = new HttpHandlerAssert(async (request) => + { + if (request.Content is not null) + { + var requestBody = await request.Content.ReadAsStringAsync().ConfigureAwait(false); + + Assert.Contains("required_tool", requestBody); + } + + return new HttpResponseMessage(HttpStatusCode.OK) { Content = new StringContent(TestDataUtil.GetAgentVersionResponseJson(), Encoding.UTF8, "application/json") }; + }); + +#pragma warning disable CA5399 + using var httpClient = new HttpClient(httpHandler); +#pragma warning restore CA5399 + + var client = new AIProjectClient(new Uri("https://test.openai.azure.com/"), new FakeAuthenticationTokenProvider(), new() { Transport = new HttpClientPipelineTransport(httpClient) }); + + // Act + var agent = await client.CreateAIAgentAsync( + name: "test-agent", + model: "test-model", + instructions: "Test", + tools: [AIFunctionFactory.Create(() => true, "required_tool")]); + + // Assert + Assert.NotNull(agent); + Assert.IsType(agent); + var agentVersion = agent.GetService(); + Assert.NotNull(agentVersion); + Assert.IsType(agentVersion.Definition); + } + + /// + /// Verify that CreateAIAgent when AI Tools are provided, uses them for the definition via http request. + /// + [Fact] + public void CreateAIAgent_WithNameAndAITools_SendsToolDefinitionViaHttp() + { + // Arrange + using var httpHandler = new HttpHandlerAssert((request) => + { + if (request.Content is not null) + { +#pragma warning disable VSTHRD002 // Avoid problematic synchronous waits + var requestBody = request.Content.ReadAsStringAsync().GetAwaiter().GetResult(); +#pragma warning restore VSTHRD002 // Avoid problematic synchronous waits + + Assert.Contains("required_tool", requestBody); + } + + return new HttpResponseMessage(HttpStatusCode.OK) { Content = new StringContent(TestDataUtil.GetAgentVersionResponseJson(), Encoding.UTF8, "application/json") }; + }); + +#pragma warning disable CA5399 + using var httpClient = new HttpClient(httpHandler); +#pragma warning restore CA5399 + + var client = new AIProjectClient(new Uri("https://test.openai.azure.com/"), new FakeAuthenticationTokenProvider(), new() { Transport = new HttpClientPipelineTransport(httpClient) }); + + // Act + var agent = client.CreateAIAgent( + name: "test-agent", + model: "test-model", + instructions: "Test", + tools: [AIFunctionFactory.Create(() => true, "required_tool")]); + + // Assert + Assert.NotNull(agent); + Assert.IsType(agent); + var agentVersion = agent.GetService(); + Assert.NotNull(agentVersion); + Assert.IsType(agentVersion.Definition); + } + + /// + /// Verify that CreateAIAgent without tools creates an agent successfully. + /// + [Fact] + public void CreateAIAgent_WithoutTools_CreatesAgentSuccessfully() + { + // Arrange + var definition = new PromptAgentDefinition("test-model"); + + var agentDefinitionResponse = GeneratePromptDefinitionResponse(definition, null); + AIProjectClient client = this.CreateTestAgentClient(agentName: "test-agent", agentDefinitionResponse: agentDefinitionResponse); + + var options = new AgentVersionCreationOptions(definition); + + // Act + var agent = client.CreateAIAgent("test-agent", options); + + // Assert + Assert.NotNull(agent); + Assert.IsType(agent); + } + + /// + /// Verify that when providing AITools with GetAIAgent, any additional tool that doesn't match the tools in agent definition are ignored. + /// + [Fact] + public void GetAIAgent_AdditionalAITools_WhenNotInTheDefinitionAreIgnored() + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(); + var agentVersion = this.CreateTestAgentVersion(); + + // Manually add tools to the definition to simulate inline tools + if (agentVersion.Definition is PromptAgentDefinition promptDef) + { + promptDef.Tools.Add(ResponseTool.CreateFunctionTool("inline_tool", BinaryData.FromString("{}"), strictModeEnabled: false)); + } + + var invocableInlineAITool = AIFunctionFactory.Create(() => "test", "inline_tool", "An invocable AIFunction for the inline function"); + var shouldBeIgnoredTool = AIFunctionFactory.Create(() => "test", "additional_tool", "An additional test function that should be ignored"); + + // Act & Assert + var agent = client.GetAIAgent(agentVersion, tools: [invocableInlineAITool, shouldBeIgnoredTool]); + Assert.NotNull(agent); + var version = agent.GetService(); + Assert.NotNull(version); + var definition = Assert.IsType(version.Definition); + Assert.NotEmpty(definition.Tools); + Assert.NotNull(GetAgentChatOptions(agent)); + Assert.NotNull(GetAgentChatOptions(agent)!.Tools); + Assert.Single(GetAgentChatOptions(agent)!.Tools!); + Assert.Equal("inline_tool", (definition.Tools.First() as FunctionTool)?.FunctionName); + } + + #endregion + + #region Inline Tools vs Parameter Tools Tests + + /// + /// Verify that tools passed as parameters are accepted by GetAIAgent. + /// + [Fact] + public void GetAIAgent_WithParameterTools_AcceptsTools() + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(); + AgentRecord agentRecord = this.CreateTestAgentRecord(); + var tools = new List + { + AIFunctionFactory.Create(() => "tool1", "param_tool_1", "First parameter tool"), + AIFunctionFactory.Create(() => "tool2", "param_tool_2", "Second parameter tool") + }; + + // Act + var agent = client.GetAIAgent(agentRecord, tools: tools); + + // Assert + Assert.NotNull(agent); + Assert.IsType(agent); + var chatClient = agent.GetService(); + Assert.NotNull(chatClient); + var agentVersion = chatClient.GetService(); + Assert.NotNull(agentVersion); + } + + /// + /// Verify that CreateAIAgent with tools in definition creates an agent successfully. + /// + [Fact] + public void CreateAIAgent_WithDefinitionTools_CreatesAgentSuccessfully() + { + // Arrange + var definition = new PromptAgentDefinition("test-model") { Instructions = "Test instructions" }; + definition.Tools.Add(ResponseTool.CreateFunctionTool("create_tool", BinaryData.FromString("{}"), strictModeEnabled: false)); + + // Simulate agent definition response with the tools + var definitionResponse = GeneratePromptDefinitionResponse(definition, definition.Tools.Select(t => t.AsAITool()).ToList()); + + AIProjectClient client = this.CreateTestAgentClient(agentDefinitionResponse: definitionResponse); + + var options = new AgentVersionCreationOptions(definition); + + // Act + var agent = client.CreateAIAgent("test-agent", options); + + // Assert + Assert.NotNull(agent); + Assert.IsType(agent); + var agentVersion = agent.GetService(); + Assert.NotNull(agentVersion); + if (agentVersion.Definition is PromptAgentDefinition promptDef) + { + Assert.NotEmpty(promptDef.Tools); + Assert.Single(promptDef.Tools); + } + } + + /// + /// Verify that CreateAIAgent creates an agent successfully when definition has a mix of custom and hosted tools. + /// + [Fact] + public void CreateAIAgent_WithMixedToolsInDefinition_CreatesAgentSuccessfully() + { + // Arrange + var definition = new PromptAgentDefinition("test-model") { Instructions = "Test instructions" }; + definition.Tools.Add(ResponseTool.CreateFunctionTool("create_tool", BinaryData.FromString("{}"), strictModeEnabled: false)); + definition.Tools.Add(new HostedWebSearchTool().GetService() ?? new HostedWebSearchTool().AsOpenAIResponseTool()); + definition.Tools.Add(new HostedFileSearchTool().GetService() ?? new HostedFileSearchTool().AsOpenAIResponseTool()); + + // Simulate agent definition response with the tools + var definitionResponse = new PromptAgentDefinition("test-model") { Instructions = "Test instructions" }; + foreach (var tool in definition.Tools) + { + definitionResponse.Tools.Add(tool); + } + + AIProjectClient client = this.CreateTestAgentClient(agentDefinitionResponse: definitionResponse); + + var options = new AgentVersionCreationOptions(definition); + + // Act + var agent = client.CreateAIAgent("test-agent", options); + + // Assert + Assert.NotNull(agent); + Assert.IsType(agent); + var agentVersion = agent.GetService(); + Assert.NotNull(agentVersion); + if (agentVersion.Definition is PromptAgentDefinition promptDef) + { + Assert.NotEmpty(promptDef.Tools); + Assert.Equal(3, promptDef.Tools.Count); + } + } + + /// + /// Verifies that CreateAIAgent uses tools from definition when they are ResponseTool instances, resulting in successful agent creation. + /// + [Fact] + public void CreateAIAgent_WithResponseToolsInDefinition_CreatesAgentSuccessfully() + { + // Arrange + var definition = new PromptAgentDefinition("test-model") { Instructions = "Test instructions" }; + + var fabricToolOptions = new FabricDataAgentToolOptions(); + fabricToolOptions.ProjectConnections.Add(new ToolProjectConnection("connection-id")); + + var sharepointOptions = new SharePointGroundingToolOptions(); + sharepointOptions.ProjectConnections.Add(new ToolProjectConnection("connection-id")); + + var structuredOutputs = new StructuredOutputDefinition("name", "description", BinaryData.FromString(AIJsonUtilities.CreateJsonSchema(new { id = "test" }.GetType()).ToString()), false); + + // Add tools to the definition + definition.Tools.Add(ResponseTool.CreateFunctionTool("create_tool", BinaryData.FromString("{}"), strictModeEnabled: false)); + definition.Tools.Add((ResponseTool)AgentTool.CreateBingCustomSearchTool(new BingCustomSearchToolParameters([new BingCustomSearchConfiguration("connection-id", "instance-name")]))); + definition.Tools.Add((ResponseTool)AgentTool.CreateBrowserAutomationTool(new BrowserAutomationToolParameters(new BrowserAutomationToolConnectionParameters("id")))); + definition.Tools.Add(AgentTool.CreateA2ATool(new Uri("https://test-uri.microsoft.com"))); + definition.Tools.Add((ResponseTool)AgentTool.CreateBingGroundingTool(new BingGroundingSearchToolOptions([new BingGroundingSearchConfiguration("connection-id")]))); + definition.Tools.Add((ResponseTool)AgentTool.CreateMicrosoftFabricTool(fabricToolOptions)); + definition.Tools.Add((ResponseTool)AgentTool.CreateOpenApiTool(new OpenAPIFunctionDefinition("name", BinaryData.FromString(OpenAPISpec), new OpenAPIAnonymousAuthenticationDetails()))); + definition.Tools.Add((ResponseTool)AgentTool.CreateSharepointTool(sharepointOptions)); + definition.Tools.Add((ResponseTool)AgentTool.CreateStructuredOutputsTool(structuredOutputs)); + definition.Tools.Add((ResponseTool)AgentTool.CreateAzureAISearchTool(new AzureAISearchToolOptions([new AzureAISearchToolIndex() { IndexName = "name" }]))); + + // Generate agent definition response with the tools + var definitionResponse = GeneratePromptDefinitionResponse(definition, definition.Tools.Select(t => t.AsAITool()).ToList()); + + AIProjectClient client = this.CreateTestAgentClient(agentDefinitionResponse: definitionResponse); + + var options = new AgentVersionCreationOptions(definition); + + // Act + var agent = client.CreateAIAgent("test-agent", options); + + // Assert + Assert.NotNull(agent); + Assert.IsType(agent); + var agentVersion = agent.GetService(); + Assert.NotNull(agentVersion); + if (agentVersion.Definition is PromptAgentDefinition promptDef) + { + Assert.NotEmpty(promptDef.Tools); + Assert.Equal(10, promptDef.Tools.Count); + } + } + + /// + /// Verify that CreateAIAgent with string parameters and tools creates an agent. + /// + [Fact] + public void CreateAIAgent_WithStringParamsAndTools_CreatesAgent() + { + // Arrange + var tools = new List + { + AIFunctionFactory.Create(() => "weather", "string_param_tool", "Tool from string params") + }; + + var definitionResponse = GeneratePromptDefinitionResponse(new PromptAgentDefinition("test-model") { Instructions = "Test instructions" }, tools); + + AIProjectClient client = this.CreateTestAgentClient(agentName: "test-agent", agentDefinitionResponse: definitionResponse); + + // Act + var agent = client.CreateAIAgent( + "test-agent", + "test-model", + "Test instructions", + tools: tools); + + // Assert + Assert.NotNull(agent); + Assert.IsType(agent); + var agentVersion = agent.GetService(); + Assert.NotNull(agentVersion); + if (agentVersion.Definition is PromptAgentDefinition promptDef) + { + Assert.NotEmpty(promptDef.Tools); + Assert.Single(promptDef.Tools); + } + } + + /// + /// Verify that CreateAIAgentAsync with tools in definition creates an agent. + /// + [Fact] + public async Task CreateAIAgentAsync_WithDefinitionTools_CreatesAgentAsync() + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(); + var definition = new PromptAgentDefinition("test-model") { Instructions = "Test instructions" }; + definition.Tools.Add(ResponseTool.CreateFunctionTool("async_tool", BinaryData.FromString("{}"), strictModeEnabled: false)); + + var options = new AgentVersionCreationOptions(definition); + + // Act + var agent = await client.CreateAIAgentAsync("test-agent", options); + + // Assert + Assert.NotNull(agent); + Assert.IsType(agent); + } + + /// + /// Verify that GetAIAgentAsync with tools parameter creates an agent. + /// + [Fact] + public async Task GetAIAgentAsync_WithToolsParameter_CreatesAgentAsync() + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(); + var tools = new List + { + AIFunctionFactory.Create(() => "async_get_result", "async_get_tool", "An async get tool") + }; + + // Act + var agent = await client.GetAIAgentAsync("test-agent", tools: tools); + + // Assert + Assert.NotNull(agent); + Assert.IsType(agent); + } + + #endregion + + #region Declarative Function Handling Tests + + /// + /// Verify that CreateAIAgent accepts declarative functions from definition. + /// + [Fact] + public void CreateAIAgent_WithDeclarativeFunctionInDefinition_AcceptsDeclarativeFunction() + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(); + var definition = new PromptAgentDefinition("test-model") { Instructions = "Test" }; + + // Create a declarative function (not invocable) using AIFunctionFactory.CreateDeclaration + using var doc = JsonDocument.Parse("{}"); + var declarativeFunction = AIFunctionFactory.CreateDeclaration("test_function", "A test function", doc.RootElement); + + // Add to definition + definition.Tools.Add(declarativeFunction.AsOpenAIResponseTool() ?? throw new InvalidOperationException()); + + var options = new AgentVersionCreationOptions(definition); + + // Act + var agent = client.CreateAIAgent("test-agent", options); + + // Assert + Assert.NotNull(agent); + Assert.IsType(agent); + } + + /// + /// Verify that CreateAIAgent accepts declarative functions from definition. + /// + [Fact] + public void CreateAIAgent_WithDeclarativeFunctionFromDefinition_AcceptsDeclarativeFunction() + { + // Arrange + var definition = new PromptAgentDefinition("test-model") { Instructions = "Test" }; + + // Create a declarative function (not invocable) using AIFunctionFactory.CreateDeclaration + using var doc = JsonDocument.Parse("{}"); + var declarativeFunction = AIFunctionFactory.CreateDeclaration("test_function", "A test function", doc.RootElement); + + // Add to definition + definition.Tools.Add(declarativeFunction.AsOpenAIResponseTool() ?? throw new InvalidOperationException()); + + // Generate response with the declarative function + var definitionResponse = new PromptAgentDefinition("test-model") { Instructions = "Test" }; + definitionResponse.Tools.Add(declarativeFunction.AsOpenAIResponseTool() ?? throw new InvalidOperationException()); + + AIProjectClient client = this.CreateTestAgentClient(agentName: "test-agent", agentDefinitionResponse: definitionResponse); + + var options = new AgentVersionCreationOptions(definition); + + // Act + var agent = client.CreateAIAgent("test-agent", options); + + // Assert + Assert.NotNull(agent); + Assert.IsType(agent); + } + + /// + /// Verify that CreateAIAgent accepts FunctionTools from definition. + /// + [Fact] + public void CreateAIAgent_WithFunctionToolsInDefinition_AcceptsDeclarativeFunction() + { + // Arrange + var functionTool = ResponseTool.CreateFunctionTool( + functionName: "get_user_name", + functionParameters: BinaryData.FromString("{}"), + strictModeEnabled: false, + functionDescription: "Gets the user's name, as used for friendly address." + ); + + var definition = new PromptAgentDefinition("test-model") { Instructions = "Test" }; + definition.Tools.Add(functionTool); + + // Generate response with the declarative function + var definitionResponse = new PromptAgentDefinition("test-model") { Instructions = "Test" }; + definitionResponse.Tools.Add(functionTool); + + AIProjectClient client = this.CreateTestAgentClient(agentName: "test-agent", agentDefinitionResponse: definitionResponse); + + var options = new AgentVersionCreationOptions(definition); + + // Act + var agent = client.CreateAIAgent("test-agent", options); + + // Assert + Assert.NotNull(agent); + Assert.IsType(agent); + var definitionFromAgent = Assert.IsType(agent.GetService()?.Definition); + Assert.Single(definitionFromAgent.Tools); + } + + /// + /// Verify that CreateAIAgentAsync accepts FunctionTools from definition. + /// + [Fact] + public async Task CreateAIAgentAsync_WithFunctionToolsInDefinition_AcceptsDeclarativeFunctionAsync() + { + // Arrange + var functionTool = ResponseTool.CreateFunctionTool( + functionName: "get_user_name", + functionParameters: BinaryData.FromString("{}"), + strictModeEnabled: false, + functionDescription: "Gets the user's name, as used for friendly address." + ); + + var definition = new PromptAgentDefinition("test-model") { Instructions = "Test" }; + definition.Tools.Add(functionTool); + + // Generate response with the declarative function + var definitionResponse = new PromptAgentDefinition("test-model") { Instructions = "Test" }; + definitionResponse.Tools.Add(functionTool); + + AIProjectClient client = this.CreateTestAgentClient(agentName: "test-agent", agentDefinitionResponse: definitionResponse); + + var options = new AgentVersionCreationOptions(definition); + + // Act + var agent = await client.CreateAIAgentAsync("test-agent", options); + + // Assert + Assert.NotNull(agent); + Assert.IsType(agent); + } + + /// + /// Verify that CreateAIAgentAsync accepts declarative functions from definition. + /// + [Fact] + public async Task CreateAIAgentAsync_WithDeclarativeFunctionFromDefinition_AcceptsDeclarativeFunctionAsync() + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(); + var definition = new PromptAgentDefinition("test-model") { Instructions = "Test" }; + + // Create a declarative function (not invocable) using AIFunctionFactory.CreateDeclaration + using var doc = JsonDocument.Parse("{}"); + var declarativeFunction = AIFunctionFactory.CreateDeclaration("test_function", "A test function", doc.RootElement); + + // Add to definition + definition.Tools.Add(declarativeFunction.AsOpenAIResponseTool() ?? throw new InvalidOperationException()); + + var options = new AgentVersionCreationOptions(definition); + + // Act + var agent = await client.CreateAIAgentAsync("test-agent", options); + + // Assert + Assert.NotNull(agent); + Assert.IsType(agent); + } + + /// + /// Verify that CreateAIAgentAsync accepts declarative functions from definition. + /// + [Fact] + public async Task CreateAIAgentAsync_WithDeclarativeFunctionInDefinition_AcceptsDeclarativeFunctionAsync() + { + // Arrange + var definition = new PromptAgentDefinition("test-model") { Instructions = "Test" }; + + // Create a declarative function (not invocable) using AIFunctionFactory.CreateDeclaration + using var doc = JsonDocument.Parse("{}"); + var declarativeFunction = AIFunctionFactory.CreateDeclaration("test_function", "A test function", doc.RootElement); + + // Add to definition + definition.Tools.Add(declarativeFunction.AsOpenAIResponseTool() ?? throw new InvalidOperationException()); + + // Generate response with the declarative function + var definitionResponse = new PromptAgentDefinition("test-model") { Instructions = "Test" }; + definitionResponse.Tools.Add(declarativeFunction.AsOpenAIResponseTool() ?? throw new InvalidOperationException()); + + AIProjectClient client = this.CreateTestAgentClient(agentName: "test-agent", agentDefinitionResponse: definitionResponse); + + var options = new AgentVersionCreationOptions(definition); + + // Act + var agent = await client.CreateAIAgentAsync("test-agent", options); + + // Assert + Assert.NotNull(agent); + Assert.IsType(agent); + } + + #endregion + + #region Options Generation Validation Tests + + /// + /// Verify that ChatClientAgentOptions are generated correctly without tools. + /// + [Fact] + public void CreateAIAgent_GeneratesCorrectChatClientAgentOptions() + { + // Arrange + var definition = new PromptAgentDefinition("test-model") { Instructions = "Test instructions" }; + + var definitionResponse = GeneratePromptDefinitionResponse(definition, null); + AIProjectClient client = this.CreateTestAgentClient(agentName: "test-agent", agentDefinitionResponse: definitionResponse); + + var options = new AgentVersionCreationOptions(definition); + + // Act + var agent = client.CreateAIAgent("test-agent", options); + + // Assert + Assert.NotNull(agent); + var agentVersion = agent.GetService(); + Assert.NotNull(agentVersion); + Assert.Equal("test-agent", agentVersion.Name); + Assert.Equal("Test instructions", (agentVersion.Definition as PromptAgentDefinition)?.Instructions); + } + + /// + /// Verify that ChatClientAgentOptions preserve custom properties from input options. + /// + [Fact] + public void GetAIAgent_WithOptions_PreservesCustomProperties() + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(agentName: "test-agent", instructions: "Custom instructions", description: "Custom description"); + var options = new ChatClientAgentOptions + { + Name = "test-agent", + Description = "Custom description", + ChatOptions = new ChatOptions { Instructions = "Custom instructions" } + }; + + // Act + var agent = client.GetAIAgent(options); + + // Assert + Assert.NotNull(agent); + Assert.Equal("test-agent", agent.Name); + Assert.Equal("Custom instructions", agent.Instructions); + Assert.Equal("Custom description", agent.Description); + } + + /// + /// Verify that CreateAIAgent with options generates correct ChatClientAgentOptions with tools. + /// + [Fact] + public void CreateAIAgent_WithOptionsAndTools_GeneratesCorrectOptions() + { + // Arrange + var tools = new List + { + AIFunctionFactory.Create(() => "result", "option_tool", "A tool from options") + }; + + var definitionResponse = GeneratePromptDefinitionResponse( + new PromptAgentDefinition("test-model") { Instructions = "Test" }, + tools); + + AIProjectClient client = this.CreateTestAgentClient(agentName: "test-agent", agentDefinitionResponse: definitionResponse); + + var options = new ChatClientAgentOptions + { + Name = "test-agent", + ChatOptions = new ChatOptions { Instructions = "Test", Tools = tools } + }; + + // Act + var agent = client.CreateAIAgent("test-model", options); + + // Assert + Assert.NotNull(agent); + var agentVersion = agent.GetService(); + Assert.NotNull(agentVersion); + if (agentVersion.Definition is PromptAgentDefinition promptDef) + { + Assert.NotEmpty(promptDef.Tools); + Assert.Single(promptDef.Tools); + } + } + + #endregion + + #region AgentName Validation Tests + + /// + /// Verify that GetAIAgent throws ArgumentException when agent name is invalid. + /// + [Theory] + [MemberData(nameof(InvalidAgentNameTestData.GetInvalidAgentNames), MemberType = typeof(InvalidAgentNameTestData))] + public void GetAIAgent_ByName_WithInvalidAgentName_ThrowsArgumentException(string invalidName) + { + // Arrange + var mockClient = new Mock(); + + // Act & Assert + var exception = Assert.Throws(() => + mockClient.Object.GetAIAgent(invalidName)); + + Assert.Equal("name", exception.ParamName); + Assert.Contains("Agent name must be 1-63 characters long", exception.Message); + } + + /// + /// Verify that GetAIAgentAsync throws ArgumentException when agent name is invalid. + /// + [Theory] + [MemberData(nameof(InvalidAgentNameTestData.GetInvalidAgentNames), MemberType = typeof(InvalidAgentNameTestData))] + public async Task GetAIAgentAsync_ByName_WithInvalidAgentName_ThrowsArgumentExceptionAsync(string invalidName) + { + // Arrange + var mockClient = new Mock(); + + // Act & Assert + var exception = await Assert.ThrowsAsync(() => + mockClient.Object.GetAIAgentAsync(invalidName)); + + Assert.Equal("name", exception.ParamName); + Assert.Contains("Agent name must be 1-63 characters long", exception.Message); + } + + /// + /// Verify that GetAIAgent with ChatClientAgentOptions throws ArgumentException when agent name is invalid. + /// + [Theory] + [MemberData(nameof(InvalidAgentNameTestData.GetInvalidAgentNames), MemberType = typeof(InvalidAgentNameTestData))] + public void GetAIAgent_WithOptions_WithInvalidAgentName_ThrowsArgumentException(string invalidName) + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(); + var options = new ChatClientAgentOptions { Name = invalidName }; + + // Act & Assert + var exception = Assert.Throws(() => + client.GetAIAgent(options)); + + Assert.Equal("name", exception.ParamName); + Assert.Contains("Agent name must be 1-63 characters long", exception.Message); + } + + /// + /// Verify that GetAIAgentAsync with ChatClientAgentOptions throws ArgumentException when agent name is invalid. + /// + [Theory] + [MemberData(nameof(InvalidAgentNameTestData.GetInvalidAgentNames), MemberType = typeof(InvalidAgentNameTestData))] + public async Task GetAIAgentAsync_WithOptions_WithInvalidAgentName_ThrowsArgumentExceptionAsync(string invalidName) + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(); + var options = new ChatClientAgentOptions { Name = invalidName }; + + // Act & Assert + var exception = await Assert.ThrowsAsync(() => + client.GetAIAgentAsync(options)); + + Assert.Equal("name", exception.ParamName); + Assert.Contains("Agent name must be 1-63 characters long", exception.Message); + } + + /// + /// Verify that CreateAIAgent throws ArgumentException when agent name is invalid. + /// + [Theory] + [MemberData(nameof(InvalidAgentNameTestData.GetInvalidAgentNames), MemberType = typeof(InvalidAgentNameTestData))] + public void CreateAIAgent_WithBasicParams_WithInvalidAgentName_ThrowsArgumentException(string invalidName) + { + // Arrange + var mockClient = new Mock(); + + // Act & Assert + var exception = Assert.Throws(() => + mockClient.Object.CreateAIAgent(invalidName, "model", "instructions")); + + Assert.Equal("name", exception.ParamName); + Assert.Contains("Agent name must be 1-63 characters long", exception.Message); + } + + /// + /// Verify that CreateAIAgentAsync throws ArgumentException when agent name is invalid. + /// + [Theory] + [MemberData(nameof(InvalidAgentNameTestData.GetInvalidAgentNames), MemberType = typeof(InvalidAgentNameTestData))] + public async Task CreateAIAgentAsync_WithBasicParams_WithInvalidAgentName_ThrowsArgumentExceptionAsync(string invalidName) + { + // Arrange + var mockClient = new Mock(); + + // Act & Assert + var exception = await Assert.ThrowsAsync(() => + mockClient.Object.CreateAIAgentAsync(invalidName, "model", "instructions")); + + Assert.Equal("name", exception.ParamName); + Assert.Contains("Agent name must be 1-63 characters long", exception.Message); + } + + /// + /// Verify that CreateAIAgent with AgentVersionCreationOptions throws ArgumentException when agent name is invalid. + /// + [Theory] + [MemberData(nameof(InvalidAgentNameTestData.GetInvalidAgentNames), MemberType = typeof(InvalidAgentNameTestData))] + public void CreateAIAgent_WithAgentDefinition_WithInvalidAgentName_ThrowsArgumentException(string invalidName) + { + // Arrange + var mockClient = new Mock(); + var definition = new PromptAgentDefinition("test-model"); + var options = new AgentVersionCreationOptions(definition); + + // Act & Assert + var exception = Assert.Throws(() => + mockClient.Object.CreateAIAgent(invalidName, options)); + + Assert.Equal("name", exception.ParamName); + Assert.Contains("Agent name must be 1-63 characters long", exception.Message); + } + + /// + /// Verify that CreateAIAgentAsync with AgentVersionCreationOptions throws ArgumentException when agent name is invalid. + /// + [Theory] + [MemberData(nameof(InvalidAgentNameTestData.GetInvalidAgentNames), MemberType = typeof(InvalidAgentNameTestData))] + public async Task CreateAIAgentAsync_WithAgentDefinition_WithInvalidAgentName_ThrowsArgumentExceptionAsync(string invalidName) + { + // Arrange + var mockClient = new Mock(); + var definition = new PromptAgentDefinition("test-model"); + var options = new AgentVersionCreationOptions(definition); + + // Act & Assert + var exception = await Assert.ThrowsAsync(() => + mockClient.Object.CreateAIAgentAsync(invalidName, options)); + + Assert.Equal("name", exception.ParamName); + Assert.Contains("Agent name must be 1-63 characters long", exception.Message); + } + + /// + /// Verify that CreateAIAgent with ChatClientAgentOptions throws ArgumentException when agent name is invalid. + /// + [Theory] + [MemberData(nameof(InvalidAgentNameTestData.GetInvalidAgentNames), MemberType = typeof(InvalidAgentNameTestData))] + public void CreateAIAgent_WithOptions_WithInvalidAgentName_ThrowsArgumentException(string invalidName) + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(); + var options = new ChatClientAgentOptions { Name = invalidName }; + + // Act & Assert + var exception = Assert.Throws(() => + client.CreateAIAgent("test-model", options)); + + Assert.Equal("name", exception.ParamName); + Assert.Contains("Agent name must be 1-63 characters long", exception.Message); + } + + /// + /// Verify that CreateAIAgentAsync with ChatClientAgentOptions throws ArgumentException when agent name is invalid. + /// + [Theory] + [MemberData(nameof(InvalidAgentNameTestData.GetInvalidAgentNames), MemberType = typeof(InvalidAgentNameTestData))] + public async Task CreateAIAgentAsync_WithOptions_WithInvalidAgentName_ThrowsArgumentExceptionAsync(string invalidName) + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(); + var options = new ChatClientAgentOptions { Name = invalidName }; + + // Act & Assert + var exception = await Assert.ThrowsAsync(() => + client.CreateAIAgentAsync("test-model", options)); + + Assert.Equal("name", exception.ParamName); + Assert.Contains("Agent name must be 1-63 characters long", exception.Message); + } + + /// + /// Verify that GetAIAgent with AgentReference throws ArgumentException when agent name is invalid. + /// + [Theory] + [MemberData(nameof(InvalidAgentNameTestData.GetInvalidAgentNames), MemberType = typeof(InvalidAgentNameTestData))] + public void GetAIAgent_WithAgentReference_WithInvalidAgentName_ThrowsArgumentException(string invalidName) + { + // Arrange + var mockClient = new Mock(); + var agentReference = new AgentReference(invalidName, "1"); + + // Act & Assert + var exception = Assert.Throws(() => + mockClient.Object.GetAIAgent(agentReference)); + + Assert.Equal("name", exception.ParamName); + Assert.Contains("Agent name must be 1-63 characters long", exception.Message); + } + + #endregion + + #region AzureAIChatClient Behavior Tests + + /// + /// Verify that the underlying chat client created by extension methods can be wrapped with clientFactory. + /// + [Fact] + public void GetAIAgent_WithClientFactory_WrapsUnderlyingChatClient() + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(); + AgentRecord agentRecord = this.CreateTestAgentRecord(); + int factoryCallCount = 0; + + // Act + var agent = client.GetAIAgent( + agentRecord, + clientFactory: (innerClient) => + { + factoryCallCount++; + return new TestChatClient(innerClient); + }); + + // Assert + Assert.NotNull(agent); + Assert.Equal(1, factoryCallCount); + var wrappedClient = agent.GetService(); + Assert.NotNull(wrappedClient); + } + + /// + /// Verify that clientFactory is called with the correct underlying chat client. + /// + [Fact] + public void CreateAIAgent_WithClientFactory_ReceivesCorrectUnderlyingClient() + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(); + var definition = new PromptAgentDefinition("test-model") { Instructions = "Test" }; + IChatClient? receivedClient = null; + + var options = new AgentVersionCreationOptions(definition); + + // Act + var agent = client.CreateAIAgent( + "test-agent", + options, + clientFactory: (innerClient) => + { + receivedClient = innerClient; + return new TestChatClient(innerClient); + }); + + // Assert + Assert.NotNull(agent); + Assert.NotNull(receivedClient); + var wrappedClient = agent.GetService(); + Assert.NotNull(wrappedClient); + } + + /// + /// Verify that multiple clientFactory calls create independent wrapped clients. + /// + [Fact] + public void GetAIAgent_MultipleCallsWithClientFactory_CreatesIndependentClients() + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(); + AgentRecord agentRecord = this.CreateTestAgentRecord(); + + // Act + var agent1 = client.GetAIAgent( + agentRecord, + clientFactory: (innerClient) => new TestChatClient(innerClient)); + + var agent2 = client.GetAIAgent( + agentRecord, + clientFactory: (innerClient) => new TestChatClient(innerClient)); + + // Assert + Assert.NotNull(agent1); + Assert.NotNull(agent2); + var client1 = agent1.GetService(); + var client2 = agent2.GetService(); + Assert.NotNull(client1); + Assert.NotNull(client2); + Assert.NotSame(client1, client2); + } + + /// + /// Verify that agent created with clientFactory maintains agent properties. + /// + [Fact] + public void CreateAIAgent_WithClientFactory_PreservesAgentProperties() + { + // Arrange + const string AgentName = "test-agent"; + const string Model = "test-model"; + const string Instructions = "Test instructions"; + AIProjectClient client = this.CreateTestAgentClient(AgentName, Instructions); + + // Act + var agent = client.CreateAIAgent( + AgentName, + Model, + Instructions, + clientFactory: (innerClient) => new TestChatClient(innerClient)); + + // Assert + Assert.NotNull(agent); + Assert.Equal(AgentName, agent.Name); + Assert.Equal(Instructions, agent.Instructions); + var wrappedClient = agent.GetService(); + Assert.NotNull(wrappedClient); + } + + /// + /// Verify that agent created with clientFactory is created successfully. + /// + [Fact] + public void CreateAIAgent_WithClientFactory_CreatesAgentSuccessfully() + { + // Arrange + var definition = new PromptAgentDefinition("test-model") { Instructions = "Test" }; + + var agentDefinitionResponse = GeneratePromptDefinitionResponse(definition, null); + AIProjectClient client = this.CreateTestAgentClient(agentName: "test-agent", agentDefinitionResponse: agentDefinitionResponse); + + var options = new AgentVersionCreationOptions(definition); + + // Act + var agent = client.CreateAIAgent( + "test-agent", + options, + clientFactory: (innerClient) => new TestChatClient(innerClient)); + + // Assert + Assert.NotNull(agent); + var wrappedClient = agent.GetService(); + Assert.NotNull(wrappedClient); + var agentVersion = agent.GetService(); + Assert.NotNull(agentVersion); + } + + #endregion + + #region User-Agent Header Tests + + /// + /// Verify that GetAIAgent(string name) passes RequestOptions to the Protocol method. + /// + [Fact] + public void GetAIAgent_WithStringName_PassesRequestOptionsToProtocol() + { + // Arrange + RequestOptions? capturedRequestOptions = null; + + var mockAgentOperations = new Mock(); + mockAgentOperations + .Setup(x => x.GetAgent(It.IsAny(), It.IsAny())) + .Callback((name, options) => capturedRequestOptions = options) + .Returns(ClientResult.FromResponse(new MockPipelineResponse(200, BinaryData.FromString(TestDataUtil.GetAgentResponseJson())))); + + var mockAgentClient = new Mock(new Uri("https://test.openai.azure.com/"), new FakeAuthenticationTokenProvider()); + mockAgentClient.SetupGet(x => x.Agents).Returns(mockAgentOperations.Object); + mockAgentClient.Setup(x => x.GetConnection(It.IsAny())).Returns(new ClientConnection("fake-connection-id", "http://localhost", ClientPipeline.Create(), CredentialKind.None)); + + // Act + var agent = mockAgentClient.Object.GetAIAgent("test-agent"); + + // Assert + Assert.NotNull(agent); + Assert.NotNull(capturedRequestOptions); + } + + /// + /// Verify that GetAIAgentAsync(string name) passes RequestOptions to the Protocol method. + /// + [Fact] + public async Task GetAIAgentAsync_WithStringName_PassesRequestOptionsToProtocolAsync() + { + // Arrange + RequestOptions? capturedRequestOptions = null; + + var mockAgentOperations = new Mock(); + mockAgentOperations + .Setup(x => x.GetAgentAsync(It.IsAny(), It.IsAny())) + .Callback((name, options) => capturedRequestOptions = options) + .Returns(Task.FromResult(ClientResult.FromResponse(new MockPipelineResponse(200, BinaryData.FromString(TestDataUtil.GetAgentResponseJson()))))); + + var mockAgentClient = new Mock(new Uri("https://test.openai.azure.com/"), new FakeAuthenticationTokenProvider()); + mockAgentClient.SetupGet(x => x.Agents).Returns(mockAgentOperations.Object); + mockAgentClient.Setup(x => x.GetConnection(It.IsAny())).Returns(new ClientConnection("fake-connection-id", "http://localhost", ClientPipeline.Create(), CredentialKind.None)); + // Act + var agent = await mockAgentClient.Object.GetAIAgentAsync("test-agent"); + + // Assert + Assert.NotNull(agent); + Assert.NotNull(capturedRequestOptions); + } + + /// + /// Verify that CreateAIAgent(string model, ChatClientAgentOptions options) passes RequestOptions to the Protocol method. + /// + [Fact] + public void CreateAIAgent_WithChatClientAgentOptions_PassesRequestOptionsToProtocol() + { + // Arrange + RequestOptions? capturedRequestOptions = null; + + var mockAgentOperations = new Mock(); + mockAgentOperations + .Setup(x => x.CreateAgentVersion(It.IsAny(), It.IsAny(), It.IsAny())) + .Callback((name, content, options) => capturedRequestOptions = options) + .Returns(ClientResult.FromResponse(new MockPipelineResponse(200, BinaryData.FromString(TestDataUtil.GetAgentVersionResponseJson())))); + + var mockAgentClient = new Mock(new Uri("https://test.openai.azure.com/"), new FakeAuthenticationTokenProvider()); + mockAgentClient.SetupGet(x => x.Agents).Returns(mockAgentOperations.Object); + mockAgentClient.Setup(x => x.GetConnection(It.IsAny())).Returns(new ClientConnection("fake-connection-id", "http://localhost", ClientPipeline.Create(), CredentialKind.None)); + + var agentOptions = new ChatClientAgentOptions { Name = "test-agent" }; + + // Act + var agent = mockAgentClient.Object.CreateAIAgent("gpt-4", agentOptions); + + // Assert + Assert.NotNull(agent); + Assert.NotNull(capturedRequestOptions); + } + + /// + /// Verify that CreateAIAgentAsync(string model, ChatClientAgentOptions options) passes RequestOptions to the Protocol method. + /// + [Fact] + public async Task CreateAIAgentAsync_WithChatClientAgentOptions_PassesRequestOptionsToProtocolAsync() + { + // Arrange + RequestOptions? capturedRequestOptions = null; + + var mockAgentOperations = new Mock(); + mockAgentOperations + .Setup(x => x.CreateAgentVersionAsync(It.IsAny(), It.IsAny(), It.IsAny())) + .Callback((name, content, options) => capturedRequestOptions = options) + .Returns(Task.FromResult(ClientResult.FromResponse(new MockPipelineResponse(200, BinaryData.FromString(TestDataUtil.GetAgentVersionResponseJson()))))); + + var mockAgentClient = new Mock(new Uri("https://test.openai.azure.com/"), new FakeAuthenticationTokenProvider()); + mockAgentClient.SetupGet(x => x.Agents).Returns(mockAgentOperations.Object); + mockAgentClient.Setup(x => x.GetConnection(It.IsAny())).Returns(new ClientConnection("fake-connection-id", "http://localhost", ClientPipeline.Create(), CredentialKind.None)); + + var agentOptions = new ChatClientAgentOptions { Name = "test-agent" }; + + // Act + var agent = await mockAgentClient.Object.CreateAIAgentAsync("gpt-4", agentOptions); + + // Assert + Assert.NotNull(agent); + Assert.NotNull(capturedRequestOptions); + } + + /// + /// Verifies that the user-agent header is added to both synchronous and asynchronous requests made by agent creation methods. + /// + [Fact] + public async Task CreateAIAgent_UserAgentHeaderAddedToRequestsAsync() + { + using var httpHandler = new HttpHandlerAssert(request => + { + Assert.Equal("POST", request.Method.Method); + Assert.Contains("MEAI", request.Headers.UserAgent.ToString()); + + return new HttpResponseMessage(HttpStatusCode.OK) { Content = new StringContent(TestDataUtil.GetAgentResponseJson(), Encoding.UTF8, "application/json") }; + }); + +#pragma warning disable CA5399 + using var httpClient = new HttpClient(httpHandler); +#pragma warning restore CA5399 + + // Arrange + var aiProjectClient = new AIProjectClient(new Uri("https://test.openai.azure.com/"), new FakeAuthenticationTokenProvider(), new() { Transport = new HttpClientPipelineTransport(httpClient) }); + + var agentOptions = new ChatClientAgentOptions { Name = "test-agent" }; + + // Act + var agent1 = aiProjectClient.CreateAIAgent("test", agentOptions); + var agent2 = await aiProjectClient.CreateAIAgentAsync("test", agentOptions); + + // Assert + Assert.NotNull(agent1); + Assert.NotNull(agent2); + } + + /// + /// Verifies that the user-agent header is added to both synchronous and asynchronous GetAIAgent requests. + /// + [Fact] + public async Task GetAIAgent_UserAgentHeaderAddedToRequestsAsync() + { + using var httpHandler = new HttpHandlerAssert(request => + { + Assert.Equal("GET", request.Method.Method); + Assert.Contains("MEAI", request.Headers.UserAgent.ToString()); + + return new HttpResponseMessage(HttpStatusCode.OK) { Content = new StringContent(TestDataUtil.GetAgentResponseJson(), Encoding.UTF8, "application/json") }; + }); + +#pragma warning disable CA5399 + using var httpClient = new HttpClient(httpHandler); +#pragma warning restore CA5399 + + // Arrange + var aiProjectClient = new AIProjectClient(new Uri("https://test.openai.azure.com/"), new FakeAuthenticationTokenProvider(), new() { Transport = new HttpClientPipelineTransport(httpClient) }); + + // Act + var agent1 = aiProjectClient.GetAIAgent("test"); + var agent2 = await aiProjectClient.GetAIAgentAsync("test"); + + // Assert + Assert.NotNull(agent1); + Assert.NotNull(agent2); + } + + #endregion + + #region GetAIAgent(AIProjectClient, AgentReference) Tests + + /// + /// Verify that GetAIAgent throws ArgumentNullException when AIProjectClient is null. + /// + [Fact] + public void GetAIAgent_WithAgentReference_WithNullClient_ThrowsArgumentNullException() + { + // Arrange + AIProjectClient? client = null; + var agentReference = new AgentReference("test-name", "1"); + + // Act & Assert + var exception = Assert.Throws(() => + client!.GetAIAgent(agentReference)); + + Assert.Equal("aiProjectClient", exception.ParamName); + } + + /// + /// Verify that GetAIAgent throws ArgumentNullException when agentReference is null. + /// + [Fact] + public void GetAIAgent_WithAgentReference_WithNullAgentReference_ThrowsArgumentNullException() + { + // Arrange + var mockClient = new Mock(); + + // Act & Assert + var exception = Assert.Throws(() => + mockClient.Object.GetAIAgent((AgentReference)null!)); + + Assert.Equal("agentReference", exception.ParamName); + } + + /// + /// Verify that GetAIAgent with AgentReference creates a valid agent. + /// + [Fact] + public void GetAIAgent_WithAgentReference_CreatesValidAgent() + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(); + var agentReference = new AgentReference("test-name", "1"); + + // Act + var agent = client.GetAIAgent(agentReference); + + // Assert + Assert.NotNull(agent); + Assert.Equal("test-name", agent.Name); + Assert.Equal("test-name:1", agent.Id); + } + + /// + /// Verify that GetAIAgent with AgentReference and clientFactory applies the factory. + /// + [Fact] + public void GetAIAgent_WithAgentReference_WithClientFactory_AppliesFactoryCorrectly() + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(); + var agentReference = new AgentReference("test-name", "1"); + TestChatClient? testChatClient = null; + + // Act + var agent = client.GetAIAgent( + agentReference, + clientFactory: (innerClient) => testChatClient = new TestChatClient(innerClient)); + + // Assert + Assert.NotNull(agent); + var retrievedTestClient = agent.GetService(); + Assert.NotNull(retrievedTestClient); + Assert.Same(testChatClient, retrievedTestClient); + } + + /// + /// Verify that GetAIAgent with AgentReference sets the agent ID correctly. + /// + [Fact] + public void GetAIAgent_WithAgentReference_SetsAgentIdCorrectly() + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(); + var agentReference = new AgentReference("test-name", "2"); + + // Act + var agent = client.GetAIAgent(agentReference); + + // Assert + Assert.NotNull(agent); + Assert.Equal("test-name:2", agent.Id); + } + + /// + /// Verify that GetAIAgent with AgentReference and tools includes the tools in ChatOptions. + /// + [Fact] + public void GetAIAgent_WithAgentReference_WithTools_IncludesToolsInChatOptions() + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(); + var agentReference = new AgentReference("test-name", "1"); + var tools = new List + { + AIFunctionFactory.Create(() => "test", "test_function", "A test function") + }; + + // Act + var agent = client.GetAIAgent(agentReference, tools: tools); + + // Assert + Assert.NotNull(agent); + var chatOptions = GetAgentChatOptions(agent); + Assert.NotNull(chatOptions); + Assert.NotNull(chatOptions.Tools); + Assert.Single(chatOptions.Tools); + } + + #endregion + + #region GetService Tests + + /// + /// Verify that GetService returns AgentRecord for agents created from AgentRecord. + /// + [Fact] + public void GetService_WithAgentRecord_ReturnsAgentRecord() + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(); + AgentRecord agentRecord = this.CreateTestAgentRecord(); + + // Act + var agent = client.GetAIAgent(agentRecord); + var retrievedRecord = agent.GetService(); + + // Assert + Assert.NotNull(retrievedRecord); + Assert.Equal(agentRecord.Id, retrievedRecord.Id); + } + + /// + /// Verify that GetService returns null for AgentRecord when agent is created from AgentReference. + /// + [Fact] + public void GetService_WithAgentReference_ReturnsNullForAgentRecord() + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(); + var agentReference = new AgentReference("test-name", "1"); + + // Act + var agent = client.GetAIAgent(agentReference); + var retrievedRecord = agent.GetService(); + + // Assert + Assert.Null(retrievedRecord); + } + + #endregion + + #region GetService Tests + + /// + /// Verify that GetService returns AgentVersion for agents created from AgentVersion. + /// + [Fact] + public void GetService_WithAgentVersion_ReturnsAgentVersion() + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(); + AgentVersion agentVersion = this.CreateTestAgentVersion(); + + // Act + var agent = client.GetAIAgent(agentVersion); + var retrievedVersion = agent.GetService(); + + // Assert + Assert.NotNull(retrievedVersion); + Assert.Equal(agentVersion.Id, retrievedVersion.Id); + } + + /// + /// Verify that GetService returns null for AgentVersion when agent is created from AgentReference. + /// + [Fact] + public void GetService_WithAgentReference_ReturnsNullForAgentVersion() + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(); + var agentReference = new AgentReference("test-name", "1"); + + // Act + var agent = client.GetAIAgent(agentReference); + var retrievedVersion = agent.GetService(); + + // Assert + Assert.Null(retrievedVersion); + } + + #endregion + + #region ChatClientMetadata Tests + + /// + /// Verify that ChatClientMetadata is properly populated for agents created from AgentRecord. + /// + [Fact] + public void ChatClientMetadata_WithAgentRecord_IsPopulatedCorrectly() + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(); + AgentRecord agentRecord = this.CreateTestAgentRecord(); + + // Act + var agent = client.GetAIAgent(agentRecord); + var metadata = agent.GetService(); + + // Assert + Assert.NotNull(metadata); + Assert.NotNull(metadata.DefaultModelId); + } + + /// + /// Verify that ChatClientMetadata.DefaultModelId is set from PromptAgentDefinition model property. + /// + [Fact] + public void ChatClientMetadata_WithPromptAgentDefinition_SetsDefaultModelIdFromModel() + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(); + var definition = new PromptAgentDefinition("gpt-4-turbo") + { + Instructions = "Test instructions" + }; + AgentRecord agentRecord = this.CreateTestAgentRecord(definition); + + // Act + var agent = client.GetAIAgent(agentRecord); + var metadata = agent.GetService(); + + // Assert + Assert.NotNull(metadata); + // The metadata should contain the model information from the agent definition + Assert.NotNull(metadata.DefaultModelId); + Assert.Equal("gpt-4-turbo", metadata.DefaultModelId); + } + + /// + /// Verify that ChatClientMetadata is properly populated for agents created from AgentVersion. + /// + [Fact] + public void ChatClientMetadata_WithAgentVersion_IsPopulatedCorrectly() + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(); + AgentVersion agentVersion = this.CreateTestAgentVersion(); + + // Act + var agent = client.GetAIAgent(agentVersion); + var metadata = agent.GetService(); + + // Assert + Assert.NotNull(metadata); + Assert.NotNull(metadata.DefaultModelId); + Assert.Equal((agentVersion.Definition as PromptAgentDefinition)!.Model, metadata.DefaultModelId); + } + + #endregion + + #region AgentReference Availability Tests + + /// + /// Verify that GetService returns AgentReference for agents created from AgentReference. + /// + [Fact] + public void GetService_WithAgentReference_ReturnsAgentReference() + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(); + var agentReference = new AgentReference("test-agent", "1.0"); + + // Act + var agent = client.GetAIAgent(agentReference); + var retrievedReference = agent.GetService(); + + // Assert + Assert.NotNull(retrievedReference); + Assert.Equal("test-agent", retrievedReference.Name); + Assert.Equal("1.0", retrievedReference.Version); + } + + /// + /// Verify that GetService returns null for AgentReference when agent is created from AgentRecord. + /// + [Fact] + public void GetService_WithAgentRecord_ReturnsAlsoAgentReference() + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(); + AgentRecord agentRecord = this.CreateTestAgentRecord(); + + // Act + var agent = client.GetAIAgent(agentRecord); + var retrievedReference = agent.GetService(); + + // Assert + Assert.NotNull(retrievedReference); + Assert.Equal(agentRecord.Name, retrievedReference.Name); + } + + /// + /// Verify that GetService returns null for AgentReference when agent is created from AgentVersion. + /// + [Fact] + public void GetService_WithAgentVersion_ReturnsAlsoAgentReference() + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(); + AgentVersion agentVersion = this.CreateTestAgentVersion(); + + // Act + var agent = client.GetAIAgent(agentVersion); + var retrievedReference = agent.GetService(); + + // Assert + Assert.NotNull(retrievedReference); + Assert.Equal(agentVersion.Name, retrievedReference.Name); + } + + /// + /// Verify that GetService returns AgentReference with correct version information. + /// + [Fact] + public void GetService_WithAgentReference_ReturnsCorrectVersionInformation() + { + // Arrange + AIProjectClient client = this.CreateTestAgentClient(); + var agentReference = new AgentReference("versioned-agent", "3.5"); + + // Act + var agent = client.GetAIAgent(agentReference); + var retrievedReference = agent.GetService(); + + // Assert + Assert.NotNull(retrievedReference); + Assert.Equal("versioned-agent", retrievedReference.Name); + Assert.Equal("3.5", retrievedReference.Version); + } + + #endregion + + #region Helper Methods + + /// + /// Creates a test AIProjectClient with fake behavior. + /// + private FakeAgentClient CreateTestAgentClient(string? agentName = null, string? instructions = null, string? description = null, AgentDefinition? agentDefinitionResponse = null) + { + return new FakeAgentClient(agentName, instructions, description, agentDefinitionResponse); + } + + /// + /// Creates a test AgentRecord for testing. + /// + private AgentRecord CreateTestAgentRecord(AgentDefinition? agentDefinition = null) + { + return ModelReaderWriter.Read(BinaryData.FromString(TestDataUtil.GetAgentResponseJson(agentDefinition: agentDefinition)))!; + } + + private const string OpenAPISpec = """ + { + "openapi": "3.0.3", + "info": { "title": "Tiny Test API", "version": "1.0.0" }, + "paths": { + "/ping": { + "get": { + "summary": "Health check", + "operationId": "getPing", + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "type": "object", + "properties": { "message": { "type": "string" } }, + "required": ["message"] + }, + "example": { "message": "pong" } + } + } + } + } + } + } + } + } + """; + + /// + /// Creates a test AgentVersion for testing. + /// + private AgentVersion CreateTestAgentVersion() + { + return ModelReaderWriter.Read(BinaryData.FromString(TestDataUtil.GetAgentVersionResponseJson()))!; + } + + /// + /// Fake AIProjectClient for testing. + /// + private sealed class FakeAgentClient : AIProjectClient + { + public FakeAgentClient(string? agentName = null, string? instructions = null, string? description = null, AgentDefinition? agentDefinitionResponse = null) + { + this.Agents = new FakeAIProjectAgentsOperations(agentName, instructions, description, agentDefinitionResponse); + } + + public override ClientConnection GetConnection(string connectionId) + { + return new ClientConnection("fake-connection-id", "http://localhost", ClientPipeline.Create(), CredentialKind.None); + } + + public override AIProjectAgentsOperations Agents { get; } + + private sealed class FakeAIProjectAgentsOperations : AIProjectAgentsOperations + { + private readonly string? _agentName; + private readonly string? _instructions; + private readonly string? _description; + private readonly AgentDefinition? _agentDefinition; + + public FakeAIProjectAgentsOperations(string? agentName = null, string? instructions = null, string? description = null, AgentDefinition? agentDefinitionResponse = null) + { + this._agentName = agentName; + this._instructions = instructions; + this._description = description; + this._agentDefinition = agentDefinitionResponse; + } + + public override ClientResult GetAgent(string agentName, RequestOptions options) + { + var responseJson = TestDataUtil.GetAgentResponseJson(this._agentName, this._agentDefinition, this._instructions, this._description); + return ClientResult.FromValue(ModelReaderWriter.Read(BinaryData.FromString(responseJson))!, new MockPipelineResponse(200, BinaryData.FromString(responseJson))); + } + + public override ClientResult GetAgent(string agentName, CancellationToken cancellationToken = default) + { + var responseJson = TestDataUtil.GetAgentResponseJson(this._agentName, this._agentDefinition, this._instructions, this._description); + return ClientResult.FromValue(ModelReaderWriter.Read(BinaryData.FromString(responseJson))!, new MockPipelineResponse(200)); + } + + public override Task GetAgentAsync(string agentName, RequestOptions options) + { + var responseJson = TestDataUtil.GetAgentResponseJson(this._agentName, this._agentDefinition, this._instructions, this._description); + return Task.FromResult(ClientResult.FromValue(ModelReaderWriter.Read(BinaryData.FromString(responseJson))!, new MockPipelineResponse(200, BinaryData.FromString(responseJson)))); + } + + public override Task> GetAgentAsync(string agentName, CancellationToken cancellationToken = default) + { + var responseJson = TestDataUtil.GetAgentResponseJson(this._agentName, this._agentDefinition, this._instructions, this._description); + return Task.FromResult(ClientResult.FromValue(ModelReaderWriter.Read(BinaryData.FromString(responseJson))!, new MockPipelineResponse(200))); + } + + public override ClientResult CreateAgentVersion(string agentName, BinaryContent content, RequestOptions? options = null) + { + var responseJson = TestDataUtil.GetAgentVersionResponseJson(this._agentName, this._agentDefinition, this._instructions, this._description); + return ClientResult.FromValue(ModelReaderWriter.Read(BinaryData.FromString(responseJson))!, new MockPipelineResponse(200, BinaryData.FromString(responseJson))); + } + + public override ClientResult CreateAgentVersion(string agentName, AgentVersionCreationOptions? options = null, CancellationToken cancellationToken = default) + { + var responseJson = TestDataUtil.GetAgentVersionResponseJson(this._agentName, this._agentDefinition, this._instructions, this._description); + return ClientResult.FromValue(ModelReaderWriter.Read(BinaryData.FromString(responseJson))!, new MockPipelineResponse(200)); + } + + public override Task CreateAgentVersionAsync(string agentName, BinaryContent content, RequestOptions? options = null) + { + var responseJson = TestDataUtil.GetAgentVersionResponseJson(this._agentName, this._agentDefinition, this._instructions, this._description); + return Task.FromResult(ClientResult.FromValue(ModelReaderWriter.Read(BinaryData.FromString(responseJson))!, new MockPipelineResponse(200, BinaryData.FromString(responseJson)))); + } + + public override Task> CreateAgentVersionAsync(string agentName, AgentVersionCreationOptions? options = null, CancellationToken cancellationToken = default) + { + var responseJson = TestDataUtil.GetAgentVersionResponseJson(this._agentName, this._agentDefinition, this._instructions, this._description); + return Task.FromResult(ClientResult.FromValue(ModelReaderWriter.Read(BinaryData.FromString(responseJson))!, new MockPipelineResponse(200))); + } + } + } + + private static PromptAgentDefinition GeneratePromptDefinitionResponse(PromptAgentDefinition inputDefinition, List? tools) + { + var definitionResponse = new PromptAgentDefinition(inputDefinition.Model) { Instructions = inputDefinition.Instructions }; + if (tools is not null) + { + foreach (var tool in tools) + { + definitionResponse.Tools.Add(tool.GetService() ?? tool.AsOpenAIResponseTool()); + } + } + + return definitionResponse; + } + + /// + /// Test custom chat client that can be used to verify clientFactory functionality. + /// + private sealed class TestChatClient : DelegatingChatClient + { + public TestChatClient(IChatClient innerClient) : base(innerClient) + { + } + } + + /// + /// Mock pipeline response for testing ClientResult wrapping. + /// + private sealed class MockPipelineResponse : PipelineResponse + { + private readonly int _status; + private readonly MockPipelineResponseHeaders _headers; + + public MockPipelineResponse(int status, BinaryData? content = null) + { + this._status = status; + this.Content = content ?? BinaryData.Empty; + this._headers = new MockPipelineResponseHeaders(); + } + + public override int Status => this._status; + + public override string ReasonPhrase => "OK"; + + public override Stream? ContentStream + { + get => null; + set { } + } + + public override BinaryData Content { get; } + + protected override PipelineResponseHeaders HeadersCore => this._headers; + + public override BinaryData BufferContent(CancellationToken cancellationToken = default) => + throw new NotSupportedException("Buffering content is not supported for mock responses."); + + public override ValueTask BufferContentAsync(CancellationToken cancellationToken = default) => + throw new NotSupportedException("Buffering content asynchronously is not supported for mock responses."); + + public override void Dispose() + { + } + + private sealed class MockPipelineResponseHeaders : PipelineResponseHeaders + { + private readonly Dictionary _headers = new(StringComparer.OrdinalIgnoreCase) + { + { "Content-Type", "application/json" }, + { "x-ms-request-id", "test-request-id" } + }; + + public override bool TryGetValue(string name, out string? value) + { + return this._headers.TryGetValue(name, out value); + } + + public override bool TryGetValues(string name, out IEnumerable? values) + { + if (this._headers.TryGetValue(name, out var value)) + { + values = [value]; + return true; + } + + values = null; + return false; + } + + public override IEnumerator> GetEnumerator() + { + return this._headers.GetEnumerator(); + } + } + } + + #endregion + + /// + /// Helper method to access internal ChatOptions property via reflection. + /// + private static ChatOptions? GetAgentChatOptions(ChatClientAgent agent) + { + if (agent is null) + { + return null; + } + + var chatOptionsProperty = typeof(ChatClientAgent).GetProperty( + "ChatOptions", + System.Reflection.BindingFlags.Public | + System.Reflection.BindingFlags.NonPublic | + System.Reflection.BindingFlags.Instance); + + return chatOptionsProperty?.GetValue(agent) as ChatOptions; + } +} + +/// +/// Provides test data for invalid agent name validation tests. +/// +internal static class InvalidAgentNameTestData +{ + /// + /// Gets a collection of invalid agent names for theory-based testing. + /// + /// Collection of invalid agent name test cases. + public static IEnumerable GetInvalidAgentNames() + { + yield return new object[] { "-agent" }; + yield return new object[] { "agent-" }; + yield return new object[] { "agent_name" }; + yield return new object[] { "agent name" }; + yield return new object[] { "agent@name" }; + yield return new object[] { "agent#name" }; + yield return new object[] { "agent$name" }; + yield return new object[] { "agent%name" }; + yield return new object[] { "agent&name" }; + yield return new object[] { "agent*name" }; + yield return new object[] { "agent.name" }; + yield return new object[] { "agent/name" }; + yield return new object[] { "agent\\name" }; + yield return new object[] { "agent:name" }; + yield return new object[] { "agent;name" }; + yield return new object[] { "agent,name" }; + yield return new object[] { "agentname" }; + yield return new object[] { "agent?name" }; + yield return new object[] { "agent!name" }; + yield return new object[] { "agent~name" }; + yield return new object[] { "agent`name" }; + yield return new object[] { "agent^name" }; + yield return new object[] { "agent|name" }; + yield return new object[] { "agent[name" }; + yield return new object[] { "agent]name" }; + yield return new object[] { "agent{name" }; + yield return new object[] { "agent}name" }; + yield return new object[] { "agent(name" }; + yield return new object[] { "agent)name" }; + yield return new object[] { "agent+name" }; + yield return new object[] { "agent=name" }; + yield return new object[] { "a" + new string('b', 63) }; + } +} diff --git a/dotnet/tests/Microsoft.Agents.AI.AzureAI.UnitTests/AzureAIProjectChatClientTests.cs b/dotnet/tests/Microsoft.Agents.AI.AzureAI.UnitTests/AzureAIProjectChatClientTests.cs new file mode 100644 index 0000000000..eee9f520b6 --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.AzureAI.UnitTests/AzureAIProjectChatClientTests.cs @@ -0,0 +1,210 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.ClientModel.Primitives; +using System.Net; +using System.Net.Http; +using System.Text; +using System.Threading.Tasks; +using Azure.AI.Projects; + +namespace Microsoft.Agents.AI.AzureAI.UnitTests; + +public class AzureAIProjectChatClientTests +{ + /// + /// Verify that when the ChatOptions has a "conv_" prefixed conversation ID, the chat client uses conversation in the http requests via the chat client + /// + [Fact] + public async Task ChatClient_UsesDefaultConversationIdAsync() + { + // Arrange + var requestTriggered = false; + using var httpHandler = new HttpHandlerAssert(async (request) => + { + if (request.RequestUri!.PathAndQuery.Contains("openai/responses")) + { + requestTriggered = true; + + // Assert + if (request.Content is not null) + { + var requestBody = await request.Content.ReadAsStringAsync().ConfigureAwait(false); + Assert.Contains("conv_12345", requestBody); + } + + return new HttpResponseMessage(HttpStatusCode.OK) { Content = new StringContent(TestDataUtil.GetOpenAIDefaultResponseJson(), Encoding.UTF8, "application/json") }; + } + + return new HttpResponseMessage(HttpStatusCode.OK) { Content = new StringContent(TestDataUtil.GetAgentResponseJson(), Encoding.UTF8, "application/json") }; + }); + +#pragma warning disable CA5399 + using var httpClient = new HttpClient(httpHandler); +#pragma warning restore CA5399 + + var client = new AIProjectClient(new Uri("https://test.openai.azure.com/"), new FakeAuthenticationTokenProvider(), new() { Transport = new HttpClientPipelineTransport(httpClient) }); + + var agent = await client.GetAIAgentAsync( + new ChatClientAgentOptions + { + Name = "test-agent", + ChatOptions = new() { Instructions = "Test instructions", ConversationId = "conv_12345" } + }); + + // Act + var thread = agent.GetNewThread(); + await agent.RunAsync("Hello", thread); + + Assert.True(requestTriggered); + var chatClientThread = Assert.IsType(thread); + Assert.Equal("conv_12345", chatClientThread.ConversationId); + } + + /// + /// Verify that when the chat client doesn't have a default "conv_" conversation id, the chat client still uses the conversation ID in HTTP requests. + /// + [Fact] + public async Task ChatClient_UsesPerRequestConversationId_WhenNoDefaultConversationIdIsProvidedAsync() + { + // Arrange + var requestTriggered = false; + using var httpHandler = new HttpHandlerAssert(async (request) => + { + if (request.RequestUri!.PathAndQuery.Contains("openai/responses")) + { + requestTriggered = true; + + // Assert + if (request.Content is not null) + { + var requestBody = await request.Content.ReadAsStringAsync().ConfigureAwait(false); + Assert.Contains("conv_12345", requestBody); + } + + return new HttpResponseMessage(HttpStatusCode.OK) { Content = new StringContent(TestDataUtil.GetOpenAIDefaultResponseJson(), Encoding.UTF8, "application/json") }; + } + + return new HttpResponseMessage(HttpStatusCode.OK) { Content = new StringContent(TestDataUtil.GetAgentResponseJson(), Encoding.UTF8, "application/json") }; + }); + +#pragma warning disable CA5399 + using var httpClient = new HttpClient(httpHandler); +#pragma warning restore CA5399 + + var client = new AIProjectClient(new Uri("https://test.openai.azure.com/"), new FakeAuthenticationTokenProvider(), new() { Transport = new HttpClientPipelineTransport(httpClient) }); + + var agent = await client.GetAIAgentAsync( + new ChatClientAgentOptions + { + Name = "test-agent", + ChatOptions = new() { Instructions = "Test instructions" }, + }); + + // Act + var thread = agent.GetNewThread(); + await agent.RunAsync("Hello", thread, options: new ChatClientAgentRunOptions() { ChatOptions = new() { ConversationId = "conv_12345" } }); + + Assert.True(requestTriggered); + var chatClientThread = Assert.IsType(thread); + Assert.Equal("conv_12345", chatClientThread.ConversationId); + } + + /// + /// Verify that even when the chat client has a default conversation id, the chat client will prioritize the per-request conversation id provided in HTTP requests. + /// + [Fact] + public async Task ChatClient_UsesPerRequestConversationId_EvenWhenDefaultConversationIdIsProvidedAsync() + { + // Arrange + var requestTriggered = false; + using var httpHandler = new HttpHandlerAssert(async (request) => + { + if (request.RequestUri!.PathAndQuery.Contains("openai/responses")) + { + requestTriggered = true; + + // Assert + if (request.Content is not null) + { + var requestBody = await request.Content.ReadAsStringAsync().ConfigureAwait(false); + Assert.Contains("conv_12345", requestBody); + } + + return new HttpResponseMessage(HttpStatusCode.OK) { Content = new StringContent(TestDataUtil.GetOpenAIDefaultResponseJson(), Encoding.UTF8, "application/json") }; + } + + return new HttpResponseMessage(HttpStatusCode.OK) { Content = new StringContent(TestDataUtil.GetAgentResponseJson(), Encoding.UTF8, "application/json") }; + }); + +#pragma warning disable CA5399 + using var httpClient = new HttpClient(httpHandler); +#pragma warning restore CA5399 + + var client = new AIProjectClient(new Uri("https://test.openai.azure.com/"), new FakeAuthenticationTokenProvider(), new() { Transport = new HttpClientPipelineTransport(httpClient) }); + + var agent = await client.GetAIAgentAsync( + new ChatClientAgentOptions + { + Name = "test-agent", + ChatOptions = new() { Instructions = "Test instructions", ConversationId = "conv_should_not_use_default" } + }); + + // Act + var thread = agent.GetNewThread(); + await agent.RunAsync("Hello", thread, options: new ChatClientAgentRunOptions() { ChatOptions = new() { ConversationId = "conv_12345" } }); + + Assert.True(requestTriggered); + var chatClientThread = Assert.IsType(thread); + Assert.Equal("conv_12345", chatClientThread.ConversationId); + } + + /// + /// Verify that when the chat client is provided without a "conv_" prefixed conversation ID, the chat client uses the previous conversation ID in HTTP requests. + /// + [Fact] + public async Task ChatClient_UsesPreviousResponseId_WhenConversationIsNotPrefixedAsConvAsync() + { + // Arrange + var requestTriggered = false; + using var httpHandler = new HttpHandlerAssert(async (request) => + { + if (request.RequestUri!.PathAndQuery.Contains("openai/responses")) + { + requestTriggered = true; + + // Assert + if (request.Content is not null) + { + var requestBody = await request.Content.ReadAsStringAsync().ConfigureAwait(false); + Assert.Contains("resp_0888a", requestBody); + } + + return new HttpResponseMessage(HttpStatusCode.OK) { Content = new StringContent(TestDataUtil.GetOpenAIDefaultResponseJson(), Encoding.UTF8, "application/json") }; + } + + return new HttpResponseMessage(HttpStatusCode.OK) { Content = new StringContent(TestDataUtil.GetAgentResponseJson(), Encoding.UTF8, "application/json") }; + }); + +#pragma warning disable CA5399 + using var httpClient = new HttpClient(httpHandler); +#pragma warning restore CA5399 + + var client = new AIProjectClient(new Uri("https://test.openai.azure.com/"), new FakeAuthenticationTokenProvider(), new() { Transport = new HttpClientPipelineTransport(httpClient) }); + + var agent = await client.GetAIAgentAsync( + new ChatClientAgentOptions + { + Name = "test-agent", + ChatOptions = new() { Instructions = "Test instructions" }, + }); + + // Act + var thread = agent.GetNewThread(); + await agent.RunAsync("Hello", thread, options: new ChatClientAgentRunOptions() { ChatOptions = new() { ConversationId = "resp_0888a" } }); + + Assert.True(requestTriggered); + var chatClientThread = Assert.IsType(thread); + Assert.Equal("resp_0888a46cbf2b1ff3006914596e05d08195a77c3f5187b769a7", chatClientThread.ConversationId); + } +} diff --git a/dotnet/tests/Microsoft.Agents.AI.AzureAI.UnitTests/FakeAuthenticationTokenProvider.cs b/dotnet/tests/Microsoft.Agents.AI.AzureAI.UnitTests/FakeAuthenticationTokenProvider.cs new file mode 100644 index 0000000000..d37ed881ff --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.AzureAI.UnitTests/FakeAuthenticationTokenProvider.cs @@ -0,0 +1,28 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.ClientModel; +using System.ClientModel.Primitives; +using System.Collections.Generic; +using System.Threading; +using System.Threading.Tasks; + +namespace Microsoft.Agents.AI.AzureAI.UnitTests; + +internal sealed class FakeAuthenticationTokenProvider : AuthenticationTokenProvider +{ + public override GetTokenOptions? CreateTokenOptions(IReadOnlyDictionary properties) + { + return new GetTokenOptions(new Dictionary()); + } + + public override AuthenticationToken GetToken(GetTokenOptions options, CancellationToken cancellationToken) + { + return new AuthenticationToken("token-value", "token-type", DateTimeOffset.UtcNow.AddHours(1)); + } + + public override ValueTask GetTokenAsync(GetTokenOptions options, CancellationToken cancellationToken) + { + return new ValueTask(this.GetToken(options, cancellationToken)); + } +} diff --git a/dotnet/tests/Microsoft.Agents.AI.AzureAI.UnitTests/HttpHandlerAssert.cs b/dotnet/tests/Microsoft.Agents.AI.AzureAI.UnitTests/HttpHandlerAssert.cs new file mode 100644 index 0000000000..3b8025ed9e --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.AzureAI.UnitTests/HttpHandlerAssert.cs @@ -0,0 +1,40 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Net.Http; +using System.Threading; +using System.Threading.Tasks; + +namespace Microsoft.Agents.AI.AzureAI.UnitTests; + +internal sealed class HttpHandlerAssert : HttpClientHandler +{ + private readonly Func? _assertion; + private readonly Func>? _assertionAsync; + + public HttpHandlerAssert(Func assertion) + { + this._assertion = assertion; + } + public HttpHandlerAssert(Func> assertionAsync) + { + this._assertionAsync = assertionAsync; + } + + protected override async Task SendAsync(HttpRequestMessage request, CancellationToken cancellationToken) + { + if (this._assertionAsync is not null) + { + return await this._assertionAsync.Invoke(request); + } + + return this._assertion!.Invoke(request); + } + +#if NET + protected override HttpResponseMessage Send(HttpRequestMessage request, CancellationToken cancellationToken) + { + return this._assertion!(request); + } +#endif +} diff --git a/dotnet/tests/Microsoft.Agents.AI.AzureAI.UnitTests/Microsoft.Agents.AI.AzureAI.UnitTests.csproj b/dotnet/tests/Microsoft.Agents.AI.AzureAI.UnitTests/Microsoft.Agents.AI.AzureAI.UnitTests.csproj new file mode 100644 index 0000000000..193a7d47da --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.AzureAI.UnitTests/Microsoft.Agents.AI.AzureAI.UnitTests.csproj @@ -0,0 +1,19 @@ + + + + + + + + + Always + + + Always + + + Always + + + + diff --git a/dotnet/tests/Microsoft.Agents.AI.AzureAI.UnitTests/TestData/AgentResponse.json b/dotnet/tests/Microsoft.Agents.AI.AzureAI.UnitTests/TestData/AgentResponse.json new file mode 100644 index 0000000000..6e93dd65c4 --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.AzureAI.UnitTests/TestData/AgentResponse.json @@ -0,0 +1,17 @@ +{ + "object": "agent", + "id": "agent_abc123", + "name": "agent_abc123", + "versions": { + "latest": { + "metadata": {}, + "object": "agent.version", + "id": "agent_abc123:1", + "name": "agent_abc123", + "version": "1", + "description": "", + "created_at": 1761771936, + "definition": "agent-definition-placeholder" + } + } +} \ No newline at end of file diff --git a/dotnet/tests/Microsoft.Agents.AI.AzureAI.UnitTests/TestData/AgentVersionResponse.json b/dotnet/tests/Microsoft.Agents.AI.AzureAI.UnitTests/TestData/AgentVersionResponse.json new file mode 100644 index 0000000000..26e5b335ca --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.AzureAI.UnitTests/TestData/AgentVersionResponse.json @@ -0,0 +1,9 @@ +{ + "object": "agent.version", + "id": "agent_abc123:1", + "name": "agent_abc123", + "version": "1", + "description": "", + "created_at": 1761771936, + "definition": "agent-definition-placeholder" +} diff --git a/dotnet/tests/Microsoft.Agents.AI.AzureAI.UnitTests/TestData/OpenAIDefaultResponse.json b/dotnet/tests/Microsoft.Agents.AI.AzureAI.UnitTests/TestData/OpenAIDefaultResponse.json new file mode 100644 index 0000000000..a270ebf4d4 --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.AzureAI.UnitTests/TestData/OpenAIDefaultResponse.json @@ -0,0 +1,68 @@ +{ + "id": "resp_0888a46cbf2b1ff3006914596e05d08195a77c3f5187b769a7", + "object": "response", + "created_at": 1762941294, + "status": "completed", + "background": false, + "billing": { + "payer": "developer" + }, + "error": null, + "incomplete_details": null, + "instructions": null, + "max_output_tokens": null, + "max_tool_calls": null, + "model": "gpt-4o-mini-2024-07-18", + "output": [ + { + "id": "msg_0888a46cbf2b1ff3006914596f814481958e8cf500a6dabbec", + "type": "message", + "status": "completed", + "content": [ + { + "type": "output_text", + "annotations": [], + "logprobs": [], + "text": "Hello! How can I assist you today?" + } + ], + "role": "assistant" + } + ], + "parallel_tool_calls": true, + "previous_response_id": null, + "prompt_cache_key": null, + "prompt_cache_retention": null, + "reasoning": { + "effort": null, + "summary": null + }, + "safety_identifier": null, + "service_tier": "default", + "store": true, + "temperature": 1.0, + "text": { + "format": { + "type": "text" + }, + "verbosity": "medium" + }, + "tool_choice": "auto", + "tools": [], + "top_logprobs": 0, + "top_p": 1.0, + "truncation": "disabled", + "usage": { + "input_tokens": 9, + "input_tokens_details": { + "cached_tokens": 0 + }, + "output_tokens": 10, + "output_tokens_details": { + "reasoning_tokens": 0 + }, + "total_tokens": 19 + }, + "user": null, + "metadata": {} +} \ No newline at end of file diff --git a/dotnet/tests/Microsoft.Agents.AI.AzureAI.UnitTests/TestDataUtil.cs b/dotnet/tests/Microsoft.Agents.AI.AzureAI.UnitTests/TestDataUtil.cs new file mode 100644 index 0000000000..c65d10de43 --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.AzureAI.UnitTests/TestDataUtil.cs @@ -0,0 +1,101 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.ClientModel.Primitives; +using System.IO; +using Azure.AI.Projects.OpenAI; + +namespace Microsoft.Agents.AI.AzureAI.UnitTests; + +/// +/// Utility class for loading and processing test data files. +/// +internal static class TestDataUtil +{ + private static readonly string s_agentResponseJson = File.ReadAllText("TestData/AgentResponse.json"); + private static readonly string s_agentVersionResponseJson = File.ReadAllText("TestData/AgentVersionResponse.json"); + private static readonly string s_openAIDefaultResponseJson = File.ReadAllText("TestData/OpenAIDefaultResponse.json"); + + private const string AgentDefinitionPlaceholder = "\"agent-definition-placeholder\""; + + private const string DefaultAgentDefinition = """ + { + "kind": "prompt", + "model": "gpt-5-mini", + "instructions": "You are a storytelling agent. You craft engaging one-line stories based on user prompts and context.", + "tools": [] + } + """; + + /// + /// Gets the agent response JSON with optional placeholder replacements applied. + /// + public static string GetAgentResponseJson(string? agentName = null, AgentDefinition? agentDefinition = null, string? instructions = null, string? description = null) + { + var json = s_agentResponseJson; + json = ApplyAgentName(json, agentName); + json = ApplyAgentDefinition(json, agentDefinition); + json = ApplyInstructions(json, instructions); + json = ApplyDescription(json, description); + return json; + } + + /// + /// Gets the agent version response JSON with optional placeholder replacements applied. + /// + public static string GetAgentVersionResponseJson(string? agentName = null, AgentDefinition? agentDefinition = null, string? instructions = null, string? description = null) + { + var json = s_agentVersionResponseJson; + json = ApplyAgentName(json, agentName); + json = ApplyAgentDefinition(json, agentDefinition); + json = ApplyInstructions(json, instructions); + json = ApplyDescription(json, description); + return json; + } + + /// + /// Gets the OpenAI default response JSON with optional placeholder replacements applied. + /// + public static string GetOpenAIDefaultResponseJson(string? agentName = null, AgentDefinition? agentDefinition = null, string? instructions = null, string? description = null) + { + var json = s_openAIDefaultResponseJson; + json = ApplyAgentName(json, agentName); + json = ApplyAgentDefinition(json, agentDefinition); + json = ApplyInstructions(json, instructions); + json = ApplyDescription(json, description); + return json; + } + + private static string ApplyAgentName(string json, string? agentName) + { + if (!string.IsNullOrEmpty(agentName)) + { + return json.Replace("\"agent_abc123\"", $"\"{agentName}\""); + } + return json; + } + + private static string ApplyAgentDefinition(string json, AgentDefinition? definition) + { + return (definition is not null) + ? json.Replace(AgentDefinitionPlaceholder, ModelReaderWriter.Write(definition).ToString()) + : json.Replace(AgentDefinitionPlaceholder, DefaultAgentDefinition); + } + + private static string ApplyInstructions(string json, string? instructions) + { + if (!string.IsNullOrEmpty(instructions)) + { + return json.Replace("You are a storytelling agent. You craft engaging one-line stories based on user prompts and context.", instructions); + } + return json; + } + + private static string ApplyDescription(string json, string? description) + { + if (!string.IsNullOrEmpty(description)) + { + return json.Replace("\"description\": \"\"", $"\"description\": \"{description}\""); + } + return json; + } +} diff --git a/dotnet/tests/Microsoft.Agents.AI.CosmosNoSql.UnitTests/.editorconfig b/dotnet/tests/Microsoft.Agents.AI.CosmosNoSql.UnitTests/.editorconfig new file mode 100644 index 0000000000..83e05f582a --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.CosmosNoSql.UnitTests/.editorconfig @@ -0,0 +1,9 @@ +# EditorConfig overrides for Cosmos DB Unit Tests +# Multi-targeting (net472 + net9.0) causes false positives for IDE0005 (unnecessary using directives) + +root = false + +[*.cs] +# Suppress IDE0005 for this project - multi-targeting causes false positives +# These using directives ARE necessary but appear unnecessary in one target framework +dotnet_diagnostic.IDE0005.severity = none diff --git a/dotnet/tests/Microsoft.Agents.AI.CosmosNoSql.UnitTests/CosmosChatMessageStoreTests.cs b/dotnet/tests/Microsoft.Agents.AI.CosmosNoSql.UnitTests/CosmosChatMessageStoreTests.cs new file mode 100644 index 0000000000..7ce0611c99 --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.CosmosNoSql.UnitTests/CosmosChatMessageStoreTests.cs @@ -0,0 +1,759 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Collections.Generic; +using System.Linq; +using System.Text.Json; +using System.Text.Json.Serialization.Metadata; +using System.Threading.Tasks; +using Azure.Core; +using Azure.Identity; +using Microsoft.Azure.Cosmos; +using Microsoft.Extensions.AI; +using Xunit; + +namespace Microsoft.Agents.AI.CosmosNoSql.UnitTests; + +/// +/// Contains tests for . +/// +/// Test Modes: +/// - Default Mode: Cleans up all test data after each test run (deletes database) +/// - Preserve Mode: Keeps containers and data for inspection in Cosmos DB Emulator Data Explorer +/// +/// To enable Preserve Mode, set environment variable: COSMOS_PRESERVE_CONTAINERS=true +/// Example: $env:COSMOS_PRESERVE_CONTAINERS="true"; dotnet test +/// +/// In Preserve Mode, you can view the data in Cosmos DB Emulator Data Explorer at: +/// https://localhost:8081/_explorer/index.html +/// Database: AgentFrameworkTests +/// Container: ChatMessages +/// +/// Environment Variable Reference: +/// | Variable | Values | Description | +/// |----------|--------|-------------| +/// | COSMOS_PRESERVE_CONTAINERS | true / false | Controls whether to preserve test data after completion | +/// +/// Usage Examples: +/// - Run all tests in preserve mode: $env:COSMOS_PRESERVE_CONTAINERS="true"; dotnet test tests/Microsoft.Agents.AI.CosmosNoSql.UnitTests/ +/// - Run specific test category in preserve mode: $env:COSMOS_PRESERVE_CONTAINERS="true"; dotnet test tests/Microsoft.Agents.AI.CosmosNoSql.UnitTests/ --filter "Category=CosmosDB" +/// - Reset to cleanup mode: $env:COSMOS_PRESERVE_CONTAINERS=""; dotnet test tests/Microsoft.Agents.AI.CosmosNoSql.UnitTests/ +/// +[Collection("CosmosDB")] +public sealed class CosmosChatMessageStoreTests : IAsyncLifetime, IDisposable +{ + // Cosmos DB Emulator connection settings + private const string EmulatorEndpoint = "https://localhost:8081"; + private const string EmulatorKey = "C2y6yDjf5/R+ob0N8A7Cgv30VRDJIWEHLM+4QDU5DE2nQ9nDuVTqobD4b8mGGyPMbIZnqyMsEcaGQy67XIw/Jw=="; + private const string TestContainerId = "ChatMessages"; + private const string HierarchicalTestContainerId = "HierarchicalChatMessages"; + // Use unique database ID per test class instance to avoid conflicts +#pragma warning disable CA1802 // Use literals where appropriate + private static readonly string s_testDatabaseId = $"AgentFrameworkTests-ChatStore-{Guid.NewGuid():N}"; +#pragma warning restore CA1802 + + private string _connectionString = string.Empty; + private bool _emulatorAvailable; + private bool _preserveContainer; + private CosmosClient? _setupClient; // Only used for test setup/cleanup + + public async Task InitializeAsync() + { + // Check environment variable to determine if we should preserve containers + // Set COSMOS_PRESERVE_CONTAINERS=true to keep containers and data for inspection + this._preserveContainer = string.Equals(Environment.GetEnvironmentVariable("COSMOS_PRESERVE_CONTAINERS"), "true", StringComparison.OrdinalIgnoreCase); + + this._connectionString = $"AccountEndpoint={EmulatorEndpoint};AccountKey={EmulatorKey}"; + + try + { + // Only create CosmosClient for test setup - the actual tests will use connection string constructors + this._setupClient = new CosmosClient(EmulatorEndpoint, EmulatorKey); + + // Test connection by attempting to create database + var databaseResponse = await this._setupClient.CreateDatabaseIfNotExistsAsync(s_testDatabaseId); + + // Create container for simple partitioning tests + await databaseResponse.Database.CreateContainerIfNotExistsAsync( + TestContainerId, + "/conversationId", + throughput: 400); + + // Create container for hierarchical partitioning tests with hierarchical partition key + var hierarchicalContainerProperties = new ContainerProperties(HierarchicalTestContainerId, ["/tenantId", "/userId", "/sessionId"]); + await databaseResponse.Database.CreateContainerIfNotExistsAsync( + hierarchicalContainerProperties, + throughput: 400); + + this._emulatorAvailable = true; + } + catch (Exception) + { + // Emulator not available, tests will be skipped + this._emulatorAvailable = false; + this._setupClient?.Dispose(); + this._setupClient = null; + } + } + + public async Task DisposeAsync() + { + if (this._setupClient != null && this._emulatorAvailable) + { + try + { + if (this._preserveContainer) + { + // Preserve mode: Don't delete the database/container, keep data for inspection + // This allows viewing data in the Cosmos DB Emulator Data Explorer + // No cleanup needed - data persists for debugging + } + else + { + // Clean mode: Delete the test database and all data + var database = this._setupClient.GetDatabase(s_testDatabaseId); + await database.DeleteAsync(); + } + } + catch (Exception ex) + { + // Ignore cleanup errors during test teardown + Console.WriteLine($"Warning: Cleanup failed: {ex.Message}"); + } + finally + { + this._setupClient.Dispose(); + } + } + } + + public void Dispose() + { + this._setupClient?.Dispose(); + GC.SuppressFinalize(this); + } + + private void SkipIfEmulatorNotAvailable() + { + // In CI: Skip if COSMOS_EMULATOR_AVAILABLE is not set to "true" + // Locally: Skip if emulator connection check failed + var ciEmulatorAvailable = string.Equals(Environment.GetEnvironmentVariable("COSMOS_EMULATOR_AVAILABLE"), "true", StringComparison.OrdinalIgnoreCase); + + Xunit.Skip.If(!ciEmulatorAvailable && !this._emulatorAvailable, "Cosmos DB Emulator is not available"); + } + + #region Constructor Tests + + [SkippableFact] + [Trait("Category", "CosmosDB")] + public void Constructor_WithConnectionString_ShouldCreateInstance() + { + // Arrange & Act + this.SkipIfEmulatorNotAvailable(); + + // Act + using var store = new CosmosChatMessageStore(this._connectionString, s_testDatabaseId, TestContainerId, "test-conversation"); + + // Assert + Assert.NotNull(store); + Assert.Equal("test-conversation", store.ConversationId); + Assert.Equal(s_testDatabaseId, store.DatabaseId); + Assert.Equal(TestContainerId, store.ContainerId); + } + + [SkippableFact] + [Trait("Category", "CosmosDB")] + public void Constructor_WithConnectionStringNoConversationId_ShouldCreateInstance() + { + // Arrange + this.SkipIfEmulatorNotAvailable(); + + // Act + using var store = new CosmosChatMessageStore(this._connectionString, s_testDatabaseId, TestContainerId); + + // Assert + Assert.NotNull(store); + Assert.NotNull(store.ConversationId); + Assert.Equal(s_testDatabaseId, store.DatabaseId); + Assert.Equal(TestContainerId, store.ContainerId); + } + + [SkippableFact] + [Trait("Category", "CosmosDB")] + public void Constructor_WithNullConnectionString_ShouldThrowArgumentException() + { + // Arrange & Act & Assert + Assert.Throws(() => + new CosmosChatMessageStore((string)null!, s_testDatabaseId, TestContainerId, "test-conversation")); + } + + [SkippableFact] + [Trait("Category", "CosmosDB")] + public void Constructor_WithEmptyConversationId_ShouldThrowArgumentException() + { + // Arrange & Act & Assert + this.SkipIfEmulatorNotAvailable(); + + Assert.Throws(() => + new CosmosChatMessageStore(this._connectionString, s_testDatabaseId, TestContainerId, "")); + } + + #endregion + + #region AddMessagesAsync Tests + + [SkippableFact] + [Trait("Category", "CosmosDB")] + public async Task AddMessagesAsync_WithSingleMessage_ShouldAddMessageAsync() + { + // Arrange + this.SkipIfEmulatorNotAvailable(); + var conversationId = Guid.NewGuid().ToString(); + using var store = new CosmosChatMessageStore(this._connectionString, s_testDatabaseId, TestContainerId, conversationId); + var message = new ChatMessage(ChatRole.User, "Hello, world!"); + + // Act + await store.AddMessagesAsync([message]); + + // Wait a moment for eventual consistency + await Task.Delay(100); + + // Assert + var messages = await store.GetMessagesAsync(); + var messageList = messages.ToList(); + + // Simple assertion - if this fails, we know the deserialization is the issue + if (messageList.Count == 0) + { + // Let's check if we can find ANY items in the container for this conversation + var directQuery = new QueryDefinition("SELECT VALUE COUNT(1) FROM c WHERE c.conversationId = @conversationId") + .WithParameter("@conversationId", conversationId); + var countIterator = this._setupClient!.GetDatabase(s_testDatabaseId).GetContainer(TestContainerId) + .GetItemQueryIterator(directQuery, requestOptions: new QueryRequestOptions + { + PartitionKey = new PartitionKey(conversationId) + }); + + var countResponse = await countIterator.ReadNextAsync(); + var count = countResponse.FirstOrDefault(); + + // Debug: Let's see what the raw query returns + var rawQuery = new QueryDefinition("SELECT * FROM c WHERE c.conversationId = @conversationId") + .WithParameter("@conversationId", conversationId); + var rawIterator = this._setupClient!.GetDatabase(s_testDatabaseId).GetContainer(TestContainerId) + .GetItemQueryIterator(rawQuery, requestOptions: new QueryRequestOptions + { + PartitionKey = new PartitionKey(conversationId) + }); + + List rawResults = []; + while (rawIterator.HasMoreResults) + { + var rawResponse = await rawIterator.ReadNextAsync(); + rawResults.AddRange(rawResponse); + } + + string rawJson = rawResults.Count > 0 ? Newtonsoft.Json.JsonConvert.SerializeObject(rawResults[0], Newtonsoft.Json.Formatting.Indented) : "null"; + Assert.Fail($"GetMessagesAsync returned 0 messages, but direct count query found {count} items for conversation {conversationId}. Raw document: {rawJson}"); + } + + Assert.Single(messageList); + Assert.Equal("Hello, world!", messageList[0].Text); + Assert.Equal(ChatRole.User, messageList[0].Role); + } + + [SkippableFact] + [Trait("Category", "CosmosDB")] + public async Task AddMessagesAsync_WithMultipleMessages_ShouldAddAllMessagesAsync() + { + // Arrange + this.SkipIfEmulatorNotAvailable(); + var conversationId = Guid.NewGuid().ToString(); + using var store = new CosmosChatMessageStore(this._connectionString, s_testDatabaseId, TestContainerId, conversationId); + var messages = new[] + { + new ChatMessage(ChatRole.User, "First message"), + new ChatMessage(ChatRole.Assistant, "Second message"), + new ChatMessage(ChatRole.User, "Third message") + }; + + // Act + await store.AddMessagesAsync(messages); + + // Assert + var retrievedMessages = await store.GetMessagesAsync(); + var messageList = retrievedMessages.ToList(); + Assert.Equal(3, messageList.Count); + Assert.Equal("First message", messageList[0].Text); + Assert.Equal("Second message", messageList[1].Text); + Assert.Equal("Third message", messageList[2].Text); + } + + #endregion + + #region GetMessagesAsync Tests + + [SkippableFact] + [Trait("Category", "CosmosDB")] + public async Task GetMessagesAsync_WithNoMessages_ShouldReturnEmptyAsync() + { + // Arrange + this.SkipIfEmulatorNotAvailable(); + using var store = new CosmosChatMessageStore(this._connectionString, s_testDatabaseId, TestContainerId, Guid.NewGuid().ToString()); + + // Act + var messages = await store.GetMessagesAsync(); + + // Assert + Assert.Empty(messages); + } + + [SkippableFact] + [Trait("Category", "CosmosDB")] + public async Task GetMessagesAsync_WithConversationIsolation_ShouldOnlyReturnMessagesForConversationAsync() + { + // Arrange + this.SkipIfEmulatorNotAvailable(); + var conversation1 = Guid.NewGuid().ToString(); + var conversation2 = Guid.NewGuid().ToString(); + + using var store1 = new CosmosChatMessageStore(this._connectionString, s_testDatabaseId, TestContainerId, conversation1); + using var store2 = new CosmosChatMessageStore(this._connectionString, s_testDatabaseId, TestContainerId, conversation2); + + await store1.AddMessagesAsync([new ChatMessage(ChatRole.User, "Message for conversation 1")]); + await store2.AddMessagesAsync([new ChatMessage(ChatRole.User, "Message for conversation 2")]); + + // Act + var messages1 = await store1.GetMessagesAsync(); + var messages2 = await store2.GetMessagesAsync(); + + // Assert + var messageList1 = messages1.ToList(); + var messageList2 = messages2.ToList(); + Assert.Single(messageList1); + Assert.Single(messageList2); + Assert.Equal("Message for conversation 1", messageList1[0].Text); + Assert.Equal("Message for conversation 2", messageList2[0].Text); + } + + #endregion + + #region Integration Tests + + [SkippableFact] + [Trait("Category", "CosmosDB")] + public async Task FullWorkflow_AddAndGet_ShouldWorkCorrectlyAsync() + { + // Arrange + this.SkipIfEmulatorNotAvailable(); + var conversationId = $"test-conversation-{Guid.NewGuid():N}"; // Use unique conversation ID + using var originalStore = new CosmosChatMessageStore(this._connectionString, s_testDatabaseId, TestContainerId, conversationId); + + var messages = new[] + { + new ChatMessage(ChatRole.System, "You are a helpful assistant."), + new ChatMessage(ChatRole.User, "Hello!"), + new ChatMessage(ChatRole.Assistant, "Hi there! How can I help you today?"), + new ChatMessage(ChatRole.User, "What's the weather like?"), + new ChatMessage(ChatRole.Assistant, "I'm sorry, I don't have access to current weather data.") + }; + + // Act 1: Add messages + await originalStore.AddMessagesAsync(messages); + + // Act 2: Verify messages were added + var retrievedMessages = await originalStore.GetMessagesAsync(); + var retrievedList = retrievedMessages.ToList(); + Assert.Equal(5, retrievedList.Count); + + // Act 3: Create new store instance for same conversation (test persistence) + using var newStore = new CosmosChatMessageStore(this._connectionString, s_testDatabaseId, TestContainerId, conversationId); + var persistedMessages = await newStore.GetMessagesAsync(); + var persistedList = persistedMessages.ToList(); + + // Assert final state + Assert.Equal(5, persistedList.Count); + Assert.Equal("You are a helpful assistant.", persistedList[0].Text); + Assert.Equal("Hello!", persistedList[1].Text); + Assert.Equal("Hi there! How can I help you today?", persistedList[2].Text); + Assert.Equal("What's the weather like?", persistedList[3].Text); + Assert.Equal("I'm sorry, I don't have access to current weather data.", persistedList[4].Text); + } + + #endregion + + #region Disposal Tests + + [SkippableFact] + [Trait("Category", "CosmosDB")] + public void Dispose_AfterUse_ShouldNotThrow() + { + // Arrange + this.SkipIfEmulatorNotAvailable(); + var store = new CosmosChatMessageStore(this._connectionString, s_testDatabaseId, TestContainerId, Guid.NewGuid().ToString()); + + // Act & Assert + store.Dispose(); // Should not throw + } + + [SkippableFact] + [Trait("Category", "CosmosDB")] + public void Dispose_MultipleCalls_ShouldNotThrow() + { + // Arrange + this.SkipIfEmulatorNotAvailable(); + var store = new CosmosChatMessageStore(this._connectionString, s_testDatabaseId, TestContainerId, Guid.NewGuid().ToString()); + + // Act & Assert + store.Dispose(); // First call + store.Dispose(); // Second call - should not throw + } + + #endregion + + #region Hierarchical Partitioning Tests + + [SkippableFact] + [Trait("Category", "CosmosDB")] + public void Constructor_WithHierarchicalConnectionString_ShouldCreateInstance() + { + // Arrange & Act + this.SkipIfEmulatorNotAvailable(); + + // Act + using var store = new CosmosChatMessageStore(this._connectionString, s_testDatabaseId, HierarchicalTestContainerId, "tenant-123", "user-456", "session-789"); + + // Assert + Assert.NotNull(store); + Assert.Equal("session-789", store.ConversationId); + Assert.Equal(s_testDatabaseId, store.DatabaseId); + Assert.Equal(HierarchicalTestContainerId, store.ContainerId); + } + + [SkippableFact] + [Trait("Category", "CosmosDB")] + public void Constructor_WithHierarchicalEndpoint_ShouldCreateInstance() + { + // Arrange & Act + this.SkipIfEmulatorNotAvailable(); + + // Act + TokenCredential credential = new DefaultAzureCredential(); + using var store = new CosmosChatMessageStore(EmulatorEndpoint, credential, s_testDatabaseId, HierarchicalTestContainerId, "tenant-123", "user-456", "session-789"); + + // Assert + Assert.NotNull(store); + Assert.Equal("session-789", store.ConversationId); + Assert.Equal(s_testDatabaseId, store.DatabaseId); + Assert.Equal(HierarchicalTestContainerId, store.ContainerId); + } + + [SkippableFact] + [Trait("Category", "CosmosDB")] + public void Constructor_WithHierarchicalCosmosClient_ShouldCreateInstance() + { + // Arrange & Act + this.SkipIfEmulatorNotAvailable(); + + using var cosmosClient = new CosmosClient(EmulatorEndpoint, EmulatorKey); + using var store = new CosmosChatMessageStore(cosmosClient, s_testDatabaseId, HierarchicalTestContainerId, "tenant-123", "user-456", "session-789"); + + // Assert + Assert.NotNull(store); + Assert.Equal("session-789", store.ConversationId); + Assert.Equal(s_testDatabaseId, store.DatabaseId); + Assert.Equal(HierarchicalTestContainerId, store.ContainerId); + } + + [SkippableFact] + [Trait("Category", "CosmosDB")] + public void Constructor_WithHierarchicalNullTenantId_ShouldThrowArgumentException() + { + // Arrange & Act & Assert + this.SkipIfEmulatorNotAvailable(); + + Assert.Throws(() => + new CosmosChatMessageStore(this._connectionString, s_testDatabaseId, TestContainerId, null!, "user-456", "session-789")); + } + + [SkippableFact] + [Trait("Category", "CosmosDB")] + public void Constructor_WithHierarchicalEmptyUserId_ShouldThrowArgumentException() + { + // Arrange & Act & Assert + this.SkipIfEmulatorNotAvailable(); + + Assert.Throws(() => + new CosmosChatMessageStore(this._connectionString, s_testDatabaseId, HierarchicalTestContainerId, "tenant-123", "", "session-789")); + } + + [SkippableFact] + [Trait("Category", "CosmosDB")] + public void Constructor_WithHierarchicalWhitespaceSessionId_ShouldThrowArgumentException() + { + // Arrange & Act & Assert + this.SkipIfEmulatorNotAvailable(); + + Assert.Throws(() => + new CosmosChatMessageStore(this._connectionString, s_testDatabaseId, HierarchicalTestContainerId, "tenant-123", "user-456", " ")); + } + + [SkippableFact] + [Trait("Category", "CosmosDB")] + public async Task AddMessagesAsync_WithHierarchicalPartitioning_ShouldAddMessageWithMetadataAsync() + { + // Arrange + this.SkipIfEmulatorNotAvailable(); + const string TenantId = "tenant-123"; + const string UserId = "user-456"; + const string SessionId = "session-789"; + // Test hierarchical partitioning constructor with connection string + using var store = new CosmosChatMessageStore(this._connectionString, s_testDatabaseId, HierarchicalTestContainerId, TenantId, UserId, SessionId); + var message = new ChatMessage(ChatRole.User, "Hello from hierarchical partitioning!"); + + // Act + await store.AddMessagesAsync([message]); + + // Wait a moment for eventual consistency + await Task.Delay(100); + + // Assert + var messages = await store.GetMessagesAsync(); + var messageList = messages.ToList(); + + Assert.Single(messageList); + Assert.Equal("Hello from hierarchical partitioning!", messageList[0].Text); + Assert.Equal(ChatRole.User, messageList[0].Role); + + // Verify that the document is stored with hierarchical partitioning metadata + var directQuery = new QueryDefinition("SELECT * FROM c WHERE c.conversationId = @conversationId AND c.type = @type") + .WithParameter("@conversationId", SessionId) + .WithParameter("@type", "ChatMessage"); + + var iterator = this._setupClient!.GetDatabase(s_testDatabaseId).GetContainer(HierarchicalTestContainerId) + .GetItemQueryIterator(directQuery, requestOptions: new QueryRequestOptions + { + PartitionKey = new PartitionKeyBuilder().Add(TenantId).Add(UserId).Add(SessionId).Build() + }); + + var response = await iterator.ReadNextAsync(); + var document = response.FirstOrDefault(); + + Assert.NotNull(document); + // The document should have hierarchical metadata + Assert.Equal(SessionId, (string)document!.conversationId); + Assert.Equal(TenantId, (string)document!.tenantId); + Assert.Equal(UserId, (string)document!.userId); + Assert.Equal(SessionId, (string)document!.sessionId); + } + + [SkippableFact] + [Trait("Category", "CosmosDB")] + public async Task AddMessagesAsync_WithHierarchicalMultipleMessages_ShouldAddAllMessagesAsync() + { + // Arrange + this.SkipIfEmulatorNotAvailable(); + const string TenantId = "tenant-batch"; + const string UserId = "user-batch"; + const string SessionId = "session-batch"; + // Test hierarchical partitioning constructor with connection string + using var store = new CosmosChatMessageStore(this._connectionString, s_testDatabaseId, HierarchicalTestContainerId, TenantId, UserId, SessionId); + var messages = new[] + { + new ChatMessage(ChatRole.User, "First hierarchical message"), + new ChatMessage(ChatRole.Assistant, "Second hierarchical message"), + new ChatMessage(ChatRole.User, "Third hierarchical message") + }; + + // Act + await store.AddMessagesAsync(messages); + + // Wait a moment for eventual consistency + await Task.Delay(100); + + // Assert + var retrievedMessages = await store.GetMessagesAsync(); + var messageList = retrievedMessages.ToList(); + + Assert.Equal(3, messageList.Count); + Assert.Equal("First hierarchical message", messageList[0].Text); + Assert.Equal("Second hierarchical message", messageList[1].Text); + Assert.Equal("Third hierarchical message", messageList[2].Text); + } + + [SkippableFact] + [Trait("Category", "CosmosDB")] + public async Task GetMessagesAsync_WithHierarchicalPartitionIsolation_ShouldIsolateMessagesByUserIdAsync() + { + // Arrange + this.SkipIfEmulatorNotAvailable(); + const string TenantId = "tenant-isolation"; + const string UserId1 = "user-1"; + const string UserId2 = "user-2"; + const string SessionId = "session-isolation"; + + // Different userIds create different hierarchical partitions, providing proper isolation + using var store1 = new CosmosChatMessageStore(this._connectionString, s_testDatabaseId, HierarchicalTestContainerId, TenantId, UserId1, SessionId); + using var store2 = new CosmosChatMessageStore(this._connectionString, s_testDatabaseId, HierarchicalTestContainerId, TenantId, UserId2, SessionId); + + // Add messages to both stores + await store1.AddMessagesAsync([new ChatMessage(ChatRole.User, "Message from user 1")]); + await store2.AddMessagesAsync([new ChatMessage(ChatRole.User, "Message from user 2")]); + + // Wait a moment for eventual consistency + await Task.Delay(100); + + // Act & Assert + var messages1 = await store1.GetMessagesAsync(); + var messageList1 = messages1.ToList(); + + var messages2 = await store2.GetMessagesAsync(); + var messageList2 = messages2.ToList(); + + // With true hierarchical partitioning, each user sees only their own messages + Assert.Single(messageList1); + Assert.Single(messageList2); + Assert.Equal("Message from user 1", messageList1[0].Text); + Assert.Equal("Message from user 2", messageList2[0].Text); + } + + [SkippableFact] + [Trait("Category", "CosmosDB")] + public async Task SerializeDeserialize_WithHierarchicalPartitioning_ShouldPreserveStateAsync() + { + // Arrange + this.SkipIfEmulatorNotAvailable(); + const string TenantId = "tenant-serialize"; + const string UserId = "user-serialize"; + const string SessionId = "session-serialize"; + + using var originalStore = new CosmosChatMessageStore(this._connectionString, s_testDatabaseId, HierarchicalTestContainerId, TenantId, UserId, SessionId); + await originalStore.AddMessagesAsync([new ChatMessage(ChatRole.User, "Test serialization message")]); + + // Act - Serialize the store state + var serializedState = originalStore.Serialize(); + + // Create a new store from the serialized state + using var cosmosClient = new CosmosClient(EmulatorEndpoint, EmulatorKey); + var serializerOptions = new JsonSerializerOptions + { + TypeInfoResolver = new DefaultJsonTypeInfoResolver() + }; + using var deserializedStore = CosmosChatMessageStore.CreateFromSerializedState(cosmosClient, serializedState, s_testDatabaseId, HierarchicalTestContainerId, serializerOptions); + + // Wait a moment for eventual consistency + await Task.Delay(100); + + // Assert - The deserialized store should have the same functionality + var messages = await deserializedStore.GetMessagesAsync(); + var messageList = messages.ToList(); + + Assert.Single(messageList); + Assert.Equal("Test serialization message", messageList[0].Text); + Assert.Equal(SessionId, deserializedStore.ConversationId); + Assert.Equal(s_testDatabaseId, deserializedStore.DatabaseId); + Assert.Equal(HierarchicalTestContainerId, deserializedStore.ContainerId); + } + + [SkippableFact] + [Trait("Category", "CosmosDB")] + public async Task HierarchicalAndSimplePartitioning_ShouldCoexistAsync() + { + // Arrange + this.SkipIfEmulatorNotAvailable(); + const string SessionId = "coexist-session"; + + // Create simple store using simple partitioning container and hierarchical store using hierarchical container + using var simpleStore = new CosmosChatMessageStore(this._connectionString, s_testDatabaseId, TestContainerId, SessionId); + using var hierarchicalStore = new CosmosChatMessageStore(this._connectionString, s_testDatabaseId, HierarchicalTestContainerId, "tenant-coexist", "user-coexist", SessionId); + + // Add messages to both + await simpleStore.AddMessagesAsync([new ChatMessage(ChatRole.User, "Simple partitioning message")]); + await hierarchicalStore.AddMessagesAsync([new ChatMessage(ChatRole.User, "Hierarchical partitioning message")]); + + // Wait a moment for eventual consistency + await Task.Delay(100); + + // Act & Assert + var simpleMessages = await simpleStore.GetMessagesAsync(); + var simpleMessageList = simpleMessages.ToList(); + + var hierarchicalMessages = await hierarchicalStore.GetMessagesAsync(); + var hierarchicalMessageList = hierarchicalMessages.ToList(); + + // Each should only see its own messages since they use different containers + Assert.Single(simpleMessageList); + Assert.Single(hierarchicalMessageList); + Assert.Equal("Simple partitioning message", simpleMessageList[0].Text); + Assert.Equal("Hierarchical partitioning message", hierarchicalMessageList[0].Text); + } + + [SkippableFact] + [Trait("Category", "CosmosDB")] + public async Task MaxMessagesToRetrieve_ShouldLimitAndReturnMostRecentAsync() + { + // Arrange + this.SkipIfEmulatorNotAvailable(); + const string ConversationId = "max-messages-test"; + + using var store = new CosmosChatMessageStore(this._connectionString, s_testDatabaseId, TestContainerId, ConversationId); + + // Add 10 messages + var messages = new List(); + for (int i = 1; i <= 10; i++) + { + messages.Add(new ChatMessage(ChatRole.User, $"Message {i}")); + await Task.Delay(10); // Small delay to ensure different timestamps + } + await store.AddMessagesAsync(messages); + + // Wait for eventual consistency + await Task.Delay(100); + + // Act - Set max to 5 and retrieve + store.MaxMessagesToRetrieve = 5; + var retrievedMessages = await store.GetMessagesAsync(); + var messageList = retrievedMessages.ToList(); + + // Assert - Should get the 5 most recent messages (6-10) in ascending order + Assert.Equal(5, messageList.Count); + Assert.Equal("Message 6", messageList[0].Text); + Assert.Equal("Message 7", messageList[1].Text); + Assert.Equal("Message 8", messageList[2].Text); + Assert.Equal("Message 9", messageList[3].Text); + Assert.Equal("Message 10", messageList[4].Text); + } + + [SkippableFact] + [Trait("Category", "CosmosDB")] + public async Task MaxMessagesToRetrieve_Null_ShouldReturnAllMessagesAsync() + { + // Arrange + this.SkipIfEmulatorNotAvailable(); + const string ConversationId = "max-messages-null-test"; + + using var store = new CosmosChatMessageStore(this._connectionString, s_testDatabaseId, TestContainerId, ConversationId); + + // Add 10 messages + var messages = new List(); + for (int i = 1; i <= 10; i++) + { + messages.Add(new ChatMessage(ChatRole.User, $"Message {i}")); + } + await store.AddMessagesAsync(messages); + + // Wait for eventual consistency + await Task.Delay(100); + + // Act - No limit set (default null) + var retrievedMessages = await store.GetMessagesAsync(); + var messageList = retrievedMessages.ToList(); + + // Assert - Should get all 10 messages + Assert.Equal(10, messageList.Count); + Assert.Equal("Message 1", messageList[0].Text); + Assert.Equal("Message 10", messageList[9].Text); + } + + #endregion +} diff --git a/dotnet/tests/Microsoft.Agents.AI.CosmosNoSql.UnitTests/CosmosCheckpointStoreTests.cs b/dotnet/tests/Microsoft.Agents.AI.CosmosNoSql.UnitTests/CosmosCheckpointStoreTests.cs new file mode 100644 index 0000000000..8f5749b187 --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.CosmosNoSql.UnitTests/CosmosCheckpointStoreTests.cs @@ -0,0 +1,454 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Linq; +using System.Text.Json; +using System.Threading.Tasks; +using Microsoft.Agents.AI.Workflows; +using Microsoft.Agents.AI.Workflows.Checkpointing; +using Microsoft.Azure.Cosmos; +using Xunit; + +namespace Microsoft.Agents.AI.CosmosNoSql.UnitTests; + +/// +/// Contains tests for . +/// +/// Test Modes: +/// - Default Mode: Cleans up all test data after each test run (deletes database) +/// - Preserve Mode: Keeps containers and data for inspection in Cosmos DB Emulator Data Explorer +/// +/// To enable Preserve Mode, set environment variable: COSMOS_PRESERVE_CONTAINERS=true +/// Example: $env:COSMOS_PRESERVE_CONTAINERS="true"; dotnet test +/// +/// In Preserve Mode, you can view the data in Cosmos DB Emulator Data Explorer at: +/// https://localhost:8081/_explorer/index.html +/// Database: AgentFrameworkTests +/// Container: Checkpoints +/// +[Collection("CosmosDB")] +public class CosmosCheckpointStoreTests : IAsyncLifetime, IDisposable +{ + // Cosmos DB Emulator connection settings + private const string EmulatorEndpoint = "https://localhost:8081"; + private const string EmulatorKey = "C2y6yDjf5/R+ob0N8A7Cgv30VRDJIWEHLM+4QDU5DE2nQ9nDuVTqobD4b8mGGyPMbIZnqyMsEcaGQy67XIw/Jw=="; + private const string TestContainerId = "Checkpoints"; + // Use unique database ID per test class instance to avoid conflicts +#pragma warning disable CA1802 // Use literals where appropriate + private static readonly string s_testDatabaseId = $"AgentFrameworkTests-CheckpointStore-{Guid.NewGuid():N}"; +#pragma warning restore CA1802 + + private string _connectionString = string.Empty; + private CosmosClient? _cosmosClient; + private Database? _database; + private bool _emulatorAvailable; + private bool _preserveContainer; + + // JsonSerializerOptions configured for .NET 9+ compatibility + private static readonly JsonSerializerOptions s_jsonOptions = CreateJsonOptions(); + + private static JsonSerializerOptions CreateJsonOptions() + { + var options = new JsonSerializerOptions(); +#if NET9_0_OR_GREATER + options.TypeInfoResolver = new System.Text.Json.Serialization.Metadata.DefaultJsonTypeInfoResolver(); +#endif + return options; + } + + public async Task InitializeAsync() + { + // Check environment variable to determine if we should preserve containers + // Set COSMOS_PRESERVE_CONTAINERS=true to keep containers and data for inspection + this._preserveContainer = string.Equals(Environment.GetEnvironmentVariable("COSMOS_PRESERVE_CONTAINERS"), "true", StringComparison.OrdinalIgnoreCase); + + this._connectionString = $"AccountEndpoint={EmulatorEndpoint};AccountKey={EmulatorKey}"; + + try + { + this._cosmosClient = new CosmosClient(EmulatorEndpoint, EmulatorKey); + + // Test connection by attempting to create database + this._database = await this._cosmosClient.CreateDatabaseIfNotExistsAsync(s_testDatabaseId); + await this._database.CreateContainerIfNotExistsAsync( + TestContainerId, + "/runId", + throughput: 400); + + this._emulatorAvailable = true; + } + catch (Exception ex) when (ex is not (OutOfMemoryException or StackOverflowException or AccessViolationException)) + { + // Emulator not available, tests will be skipped + this._emulatorAvailable = false; + this._cosmosClient?.Dispose(); + this._cosmosClient = null; + } + } + + public async Task DisposeAsync() + { + if (this._cosmosClient != null && this._emulatorAvailable) + { + try + { + if (this._preserveContainer) + { + // Preserve mode: Don't delete the database/container, keep data for inspection + // This allows viewing data in the Cosmos DB Emulator Data Explorer + // No cleanup needed - data persists for debugging + } + else + { + // Clean mode: Delete the test database and all data + await this._database!.DeleteAsync(); + } + } + catch (Exception ex) + { + // Ignore cleanup errors, but log for diagnostics + Console.WriteLine($"[DisposeAsync] Cleanup error: {ex.Message}\n{ex.StackTrace}"); + } + finally + { + this._cosmosClient.Dispose(); + } + } + } + + private void SkipIfEmulatorNotAvailable() + { + // In CI: Skip if COSMOS_EMULATOR_AVAILABLE is not set to "true" + // Locally: Skip if emulator connection check failed + var ciEmulatorAvailable = string.Equals(Environment.GetEnvironmentVariable("COSMOS_EMULATOR_AVAILABLE"), "true", StringComparison.OrdinalIgnoreCase); + + Xunit.Skip.If(!ciEmulatorAvailable && !this._emulatorAvailable, "Cosmos DB Emulator is not available"); + } + + #region Constructor Tests + + [SkippableFact] + public void Constructor_WithCosmosClient_SetsProperties() + { + // Arrange + this.SkipIfEmulatorNotAvailable(); + + // Act + using var store = new CosmosCheckpointStore(this._cosmosClient!, s_testDatabaseId, TestContainerId); + + // Assert + Assert.Equal(s_testDatabaseId, store.DatabaseId); + Assert.Equal(TestContainerId, store.ContainerId); + } + + [SkippableFact] + public void Constructor_WithConnectionString_SetsProperties() + { + // Arrange + this.SkipIfEmulatorNotAvailable(); + + // Act + using var store = new CosmosCheckpointStore(this._connectionString, s_testDatabaseId, TestContainerId); + + // Assert + Assert.Equal(s_testDatabaseId, store.DatabaseId); + Assert.Equal(TestContainerId, store.ContainerId); + } + + [SkippableFact] + public void Constructor_WithNullCosmosClient_ThrowsArgumentNullException() + { + // Act & Assert + Assert.Throws(() => + new CosmosCheckpointStore((CosmosClient)null!, s_testDatabaseId, TestContainerId)); + } + + [SkippableFact] + public void Constructor_WithNullConnectionString_ThrowsArgumentException() + { + // Act & Assert + Assert.Throws(() => + new CosmosCheckpointStore((string)null!, s_testDatabaseId, TestContainerId)); + } + + #endregion + + #region Checkpoint Operations Tests + + [SkippableFact] + public async Task CreateCheckpointAsync_NewCheckpoint_CreatesSuccessfullyAsync() + { + this.SkipIfEmulatorNotAvailable(); + + // Arrange + using var store = new CosmosCheckpointStore(this._cosmosClient!, s_testDatabaseId, TestContainerId); + var runId = Guid.NewGuid().ToString(); + var checkpointValue = JsonSerializer.SerializeToElement(new { data = "test checkpoint" }, s_jsonOptions); + + // Act + var checkpointInfo = await store.CreateCheckpointAsync(runId, checkpointValue); + + // Assert + Assert.NotNull(checkpointInfo); + Assert.Equal(runId, checkpointInfo.RunId); + Assert.NotNull(checkpointInfo.CheckpointId); + Assert.NotEmpty(checkpointInfo.CheckpointId); + } + + [SkippableFact] + public async Task RetrieveCheckpointAsync_ExistingCheckpoint_ReturnsCorrectValueAsync() + { + this.SkipIfEmulatorNotAvailable(); + + // Arrange + using var store = new CosmosCheckpointStore(this._cosmosClient!, s_testDatabaseId, TestContainerId); + var runId = Guid.NewGuid().ToString(); + var originalData = new { message = "Hello, World!", timestamp = DateTimeOffset.UtcNow }; + var checkpointValue = JsonSerializer.SerializeToElement(originalData, s_jsonOptions); + + // Act + var checkpointInfo = await store.CreateCheckpointAsync(runId, checkpointValue); + var retrievedValue = await store.RetrieveCheckpointAsync(runId, checkpointInfo); + + // Assert + Assert.Equal(JsonValueKind.Object, retrievedValue.ValueKind); + Assert.True(retrievedValue.TryGetProperty("message", out var messageProp)); + Assert.Equal("Hello, World!", messageProp.GetString()); + } + + [SkippableFact] + public async Task RetrieveCheckpointAsync_NonExistentCheckpoint_ThrowsInvalidOperationExceptionAsync() + { + this.SkipIfEmulatorNotAvailable(); + + // Arrange + using var store = new CosmosCheckpointStore(this._cosmosClient!, s_testDatabaseId, TestContainerId); + var runId = Guid.NewGuid().ToString(); + var fakeCheckpointInfo = new CheckpointInfo(runId, "nonexistent-checkpoint"); + + // Act & Assert + await Assert.ThrowsAsync(() => + store.RetrieveCheckpointAsync(runId, fakeCheckpointInfo).AsTask()); + } + + [SkippableFact] + public async Task RetrieveIndexAsync_EmptyStore_ReturnsEmptyCollectionAsync() + { + this.SkipIfEmulatorNotAvailable(); + + // Arrange + using var store = new CosmosCheckpointStore(this._cosmosClient!, s_testDatabaseId, TestContainerId); + var runId = Guid.NewGuid().ToString(); + + // Act + var index = await store.RetrieveIndexAsync(runId); + + // Assert + Assert.NotNull(index); + Assert.Empty(index); + } + + [SkippableFact] + public async Task RetrieveIndexAsync_WithCheckpoints_ReturnsAllCheckpointsAsync() + { + this.SkipIfEmulatorNotAvailable(); + + // Arrange + using var store = new CosmosCheckpointStore(this._cosmosClient!, s_testDatabaseId, TestContainerId); + var runId = Guid.NewGuid().ToString(); + var checkpointValue = JsonSerializer.SerializeToElement(new { data = "test" }, s_jsonOptions); + + // Create multiple checkpoints + var checkpoint1 = await store.CreateCheckpointAsync(runId, checkpointValue); + var checkpoint2 = await store.CreateCheckpointAsync(runId, checkpointValue); + var checkpoint3 = await store.CreateCheckpointAsync(runId, checkpointValue); + + // Act + var index = (await store.RetrieveIndexAsync(runId)).ToList(); + + // Assert + Assert.Equal(3, index.Count); + Assert.Contains(index, c => c.CheckpointId == checkpoint1.CheckpointId); + Assert.Contains(index, c => c.CheckpointId == checkpoint2.CheckpointId); + Assert.Contains(index, c => c.CheckpointId == checkpoint3.CheckpointId); + } + + [SkippableFact] + public async Task CreateCheckpointAsync_WithParent_CreatesHierarchyAsync() + { + this.SkipIfEmulatorNotAvailable(); + + // Arrange + using var store = new CosmosCheckpointStore(this._cosmosClient!, s_testDatabaseId, TestContainerId); + var runId = Guid.NewGuid().ToString(); + var checkpointValue = JsonSerializer.SerializeToElement(new { data = "test" }, s_jsonOptions); + + // Act + var parentCheckpoint = await store.CreateCheckpointAsync(runId, checkpointValue); + var childCheckpoint = await store.CreateCheckpointAsync(runId, checkpointValue, parentCheckpoint); + + // Assert + Assert.NotEqual(parentCheckpoint.CheckpointId, childCheckpoint.CheckpointId); + Assert.Equal(runId, parentCheckpoint.RunId); + Assert.Equal(runId, childCheckpoint.RunId); + } + + [SkippableFact] + public async Task RetrieveIndexAsync_WithParentFilter_ReturnsFilteredResultsAsync() + { + this.SkipIfEmulatorNotAvailable(); + + // Arrange + using var store = new CosmosCheckpointStore(this._cosmosClient!, s_testDatabaseId, TestContainerId); + var runId = Guid.NewGuid().ToString(); + var checkpointValue = JsonSerializer.SerializeToElement(new { data = "test" }, s_jsonOptions); + + // Create parent and child checkpoints + var parent = await store.CreateCheckpointAsync(runId, checkpointValue); + var child1 = await store.CreateCheckpointAsync(runId, checkpointValue, parent); + var child2 = await store.CreateCheckpointAsync(runId, checkpointValue, parent); + + // Create an orphan checkpoint + var orphan = await store.CreateCheckpointAsync(runId, checkpointValue); + + // Act + var allCheckpoints = (await store.RetrieveIndexAsync(runId)).ToList(); + var childrenOfParent = (await store.RetrieveIndexAsync(runId, parent)).ToList(); + + // Assert + Assert.Equal(4, allCheckpoints.Count); // parent + 2 children + orphan + Assert.Equal(2, childrenOfParent.Count); // only children + + Assert.Contains(childrenOfParent, c => c.CheckpointId == child1.CheckpointId); + Assert.Contains(childrenOfParent, c => c.CheckpointId == child2.CheckpointId); + Assert.DoesNotContain(childrenOfParent, c => c.CheckpointId == parent.CheckpointId); + Assert.DoesNotContain(childrenOfParent, c => c.CheckpointId == orphan.CheckpointId); + } + + #endregion + + #region Run Isolation Tests + + [SkippableFact] + public async Task CheckpointOperations_DifferentRuns_IsolatesDataAsync() + { + this.SkipIfEmulatorNotAvailable(); + + // Arrange + using var store = new CosmosCheckpointStore(this._cosmosClient!, s_testDatabaseId, TestContainerId); + var runId1 = Guid.NewGuid().ToString(); + var runId2 = Guid.NewGuid().ToString(); + var checkpointValue = JsonSerializer.SerializeToElement(new { data = "test" }, s_jsonOptions); + + // Act + var checkpoint1 = await store.CreateCheckpointAsync(runId1, checkpointValue); + var checkpoint2 = await store.CreateCheckpointAsync(runId2, checkpointValue); + + var index1 = (await store.RetrieveIndexAsync(runId1)).ToList(); + var index2 = (await store.RetrieveIndexAsync(runId2)).ToList(); + + // Assert + Assert.Single(index1); + Assert.Single(index2); + Assert.Equal(checkpoint1.CheckpointId, index1[0].CheckpointId); + Assert.Equal(checkpoint2.CheckpointId, index2[0].CheckpointId); + Assert.NotEqual(checkpoint1.CheckpointId, checkpoint2.CheckpointId); + } + + #endregion + + #region Error Handling Tests + + [SkippableFact] + public async Task CreateCheckpointAsync_WithNullRunId_ThrowsArgumentExceptionAsync() + { + this.SkipIfEmulatorNotAvailable(); + + // Arrange + using var store = new CosmosCheckpointStore(this._cosmosClient!, s_testDatabaseId, TestContainerId); + var checkpointValue = JsonSerializer.SerializeToElement(new { data = "test" }, s_jsonOptions); + + // Act & Assert + await Assert.ThrowsAsync(() => + store.CreateCheckpointAsync(null!, checkpointValue).AsTask()); + } + + [SkippableFact] + public async Task CreateCheckpointAsync_WithEmptyRunId_ThrowsArgumentExceptionAsync() + { + this.SkipIfEmulatorNotAvailable(); + + // Arrange + using var store = new CosmosCheckpointStore(this._cosmosClient!, s_testDatabaseId, TestContainerId); + var checkpointValue = JsonSerializer.SerializeToElement(new { data = "test" }, s_jsonOptions); + + // Act & Assert + await Assert.ThrowsAsync(() => + store.CreateCheckpointAsync("", checkpointValue).AsTask()); + } + + [SkippableFact] + public async Task RetrieveCheckpointAsync_WithNullCheckpointInfo_ThrowsArgumentNullExceptionAsync() + { + this.SkipIfEmulatorNotAvailable(); + + // Arrange + using var store = new CosmosCheckpointStore(this._cosmosClient!, s_testDatabaseId, TestContainerId); + var runId = Guid.NewGuid().ToString(); + + // Act & Assert + await Assert.ThrowsAsync(() => + store.RetrieveCheckpointAsync(runId, null!).AsTask()); + } + + #endregion + + #region Disposal Tests + + [SkippableFact] + public async Task Dispose_AfterDisposal_ThrowsObjectDisposedExceptionAsync() + { + this.SkipIfEmulatorNotAvailable(); + + // Arrange + var store = new CosmosCheckpointStore(this._cosmosClient!, s_testDatabaseId, TestContainerId); + var checkpointValue = JsonSerializer.SerializeToElement(new { data = "test" }, s_jsonOptions); + + // Act + store.Dispose(); + + // Assert + await Assert.ThrowsAsync(() => + store.CreateCheckpointAsync("test-run", checkpointValue).AsTask()); + } + + [SkippableFact] + public void Dispose_MultipleCalls_DoesNotThrow() + { + this.SkipIfEmulatorNotAvailable(); + + // Arrange + var store = new CosmosCheckpointStore(this._cosmosClient!, s_testDatabaseId, TestContainerId); + + // Act & Assert (should not throw) + store.Dispose(); + store.Dispose(); + store.Dispose(); + } + + #endregion + + public void Dispose() + { + this.Dispose(true); + GC.SuppressFinalize(this); + } + + protected virtual void Dispose(bool disposing) + { + if (disposing) + { + this._cosmosClient?.Dispose(); + } + } +} diff --git a/dotnet/tests/Microsoft.Agents.AI.CosmosNoSql.UnitTests/CosmosDBCollectionFixture.cs b/dotnet/tests/Microsoft.Agents.AI.CosmosNoSql.UnitTests/CosmosDBCollectionFixture.cs new file mode 100644 index 0000000000..195c433de5 --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.CosmosNoSql.UnitTests/CosmosDBCollectionFixture.cs @@ -0,0 +1,18 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Xunit; + +namespace Microsoft.Agents.AI.CosmosNoSql.UnitTests; + +/// +/// Defines a collection fixture for Cosmos DB tests to ensure they run sequentially. +/// This prevents race conditions and resource conflicts when tests create and delete +/// databases in the Cosmos DB Emulator. +/// +[CollectionDefinition("CosmosDB", DisableParallelization = true)] +public sealed class CosmosDBCollectionFixture +{ + // This class has no code, and is never created. Its purpose is simply + // to be the place to apply [CollectionDefinition] and all the + // ICollectionFixture<> interfaces. +} diff --git a/dotnet/tests/Microsoft.Agents.AI.CosmosNoSql.UnitTests/Microsoft.Agents.AI.CosmosNoSql.UnitTests.csproj b/dotnet/tests/Microsoft.Agents.AI.CosmosNoSql.UnitTests/Microsoft.Agents.AI.CosmosNoSql.UnitTests.csproj new file mode 100644 index 0000000000..d60418ee2c --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.CosmosNoSql.UnitTests/Microsoft.Agents.AI.CosmosNoSql.UnitTests.csproj @@ -0,0 +1,24 @@ + + + + net10.0;net9.0 + $(NoWarn);MEAI001 + + + + false + + + + + + + + + + + + + + + diff --git a/dotnet/tests/Microsoft.Agents.AI.Declarative.UnitTests/AgentBotElementYamlTests.cs b/dotnet/tests/Microsoft.Agents.AI.Declarative.UnitTests/AgentBotElementYamlTests.cs new file mode 100644 index 0000000000..31cadfb0ce --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.Declarative.UnitTests/AgentBotElementYamlTests.cs @@ -0,0 +1,310 @@ +// Copyright (c) Microsoft. All rights reserved. +using System.Collections.Generic; +using System.ComponentModel; +using System.IO; +using System.Linq; +using System.Text.Json.Serialization; +using Microsoft.Bot.ObjectModel; +using Microsoft.Extensions.AI; +using Microsoft.Extensions.Configuration; +using Microsoft.PowerFx; + +namespace Microsoft.Agents.AI.Declarative.UnitTests; + +/// +/// Unit tests for +/// +public sealed class AgentBotElementYamlTests +{ + [Theory] + [InlineData(PromptAgents.AgentWithEverything)] + [InlineData(PromptAgents.AgentWithApiKeyConnection)] + [InlineData(PromptAgents.AgentWithVariableReferences)] + [InlineData(PromptAgents.AgentWithOutputSchema)] + [InlineData(PromptAgents.OpenAIChatAgent)] + [InlineData(PromptAgents.AgentWithCurrentModels)] + [InlineData(PromptAgents.AgentWithRemoteConnection)] + public void FromYaml_DoesNotThrow(string text) + { + // Arrange & Act + var agent = AgentBotElementYaml.FromYaml(text); + + // Assert + Assert.NotNull(agent); + } + + [Fact] + public void FromYaml_NotPromptAgent_Throws() + { + // Arrange & Act & Assert + Assert.Throws(() => AgentBotElementYaml.FromYaml(PromptAgents.Workflow)); + } + + [Fact] + public void FromYaml_Properties() + { + // Arrange & Act + var agent = AgentBotElementYaml.FromYaml(PromptAgents.AgentWithEverything); + + // Assert + Assert.NotNull(agent); + Assert.Equal("AgentName", agent.Name); + Assert.Equal("Agent description", agent.Description); + Assert.Equal("You are a helpful assistant.", agent.Instructions?.ToTemplateString()); + Assert.NotNull(agent.Model); + Assert.True(agent.Tools.Length > 0); + } + + [Fact] + public void FromYaml_CurrentModels() + { + // Arrange & Act + var agent = AgentBotElementYaml.FromYaml(PromptAgents.AgentWithCurrentModels); + + // Assert + Assert.NotNull(agent); + Assert.NotNull(agent.Model); + Assert.Equal("gpt-4o", agent.Model.ModelNameHint); + Assert.NotNull(agent.Model.Options); + Assert.Equal(0.7f, (float?)agent.Model.Options?.Temperature?.LiteralValue); + Assert.Equal(0.9f, (float?)agent.Model.Options?.TopP?.LiteralValue); + + // Assert contents using extension methods + Assert.Equal(1024, agent.Model.Options?.MaxOutputTokens?.LiteralValue); + Assert.Equal(50, agent.Model.Options?.TopK?.LiteralValue); + Assert.Equal(0.7f, (float?)agent.Model.Options?.FrequencyPenalty?.LiteralValue); + Assert.Equal(0.7f, (float?)agent.Model.Options?.PresencePenalty?.LiteralValue); + Assert.Equal(42, agent.Model.Options?.Seed?.LiteralValue); + Assert.Equal(PromptAgents.s_stopSequences, agent.Model.Options?.StopSequences); + Assert.True(agent.Model.Options?.AllowMultipleToolCalls?.LiteralValue); + Assert.Equal(ChatToolMode.Auto, agent.Model.Options?.AsChatToolMode()); + } + + [Fact] + public void FromYaml_OutputSchema() + { + // Arrange & Act + var agent = AgentBotElementYaml.FromYaml(PromptAgents.AgentWithOutputSchema); + + // Assert + Assert.NotNull(agent); + Assert.NotNull(agent.OutputType); + ChatResponseFormatJson responseFormat = (agent.OutputType.AsChatResponseFormat() as ChatResponseFormatJson)!; + Assert.NotNull(responseFormat); + Assert.NotNull(responseFormat.Schema); + } + + [Fact] + public void FromYaml_CodeInterpreter() + { + // Arrange & Act + var agent = AgentBotElementYaml.FromYaml(PromptAgents.AgentWithEverything); + + // Assert + Assert.NotNull(agent); + var tools = agent.Tools; + var codeInterpreterTools = tools.Where(t => t is CodeInterpreterTool).ToArray(); + Assert.Single(codeInterpreterTools); + CodeInterpreterTool codeInterpreterTool = (codeInterpreterTools[0] as CodeInterpreterTool)!; + Assert.NotNull(codeInterpreterTool); + } + + [Fact] + public void FromYaml_FunctionTool() + { + // Arrange & Act + var agent = AgentBotElementYaml.FromYaml(PromptAgents.AgentWithEverything); + + // Assert + Assert.NotNull(agent); + var tools = agent.Tools; + var functionTools = tools.Where(t => t is InvokeClientTaskAction).ToArray(); + Assert.Single(functionTools); + InvokeClientTaskAction functionTool = (functionTools[0] as InvokeClientTaskAction)!; + Assert.NotNull(functionTool); + Assert.Equal("GetWeather", functionTool.Name); + Assert.Equal("Get the weather for a given location.", functionTool.Description); + // TODO check schema + } + + [Fact] + public void FromYaml_MCP() + { + // Arrange & Act + var agent = AgentBotElementYaml.FromYaml(PromptAgents.AgentWithEverything); + + // Assert + Assert.NotNull(agent); + var tools = agent.Tools; + var mcpTools = tools.Where(t => t is McpServerTool).ToArray(); + Assert.Single(mcpTools); + McpServerTool mcpTool = (mcpTools[0] as McpServerTool)!; + Assert.NotNull(mcpTool); + Assert.Equal("PersonInfoTool", mcpTool.ServerName?.LiteralValue); + AnonymousConnection connection = (mcpTool.Connection as AnonymousConnection)!; + Assert.NotNull(connection); + Assert.Equal("https://my-mcp-endpoint.com/api", connection.Endpoint?.LiteralValue); + } + + [Fact] + public void FromYaml_WebSearchTool() + { + // Arrange & Act + var agent = AgentBotElementYaml.FromYaml(PromptAgents.AgentWithEverything); + + // Assert + Assert.NotNull(agent); + var tools = agent.Tools; + var webSearchTools = tools.Where(t => t is WebSearchTool).ToArray(); + Assert.Single(webSearchTools); + Assert.NotNull(webSearchTools[0] as WebSearchTool); + } + + [Fact] + public void FromYaml_FileSearchTool() + { + // Arrange & Act + var agent = AgentBotElementYaml.FromYaml(PromptAgents.AgentWithEverything); + + // Assert + Assert.NotNull(agent); + var tools = agent.Tools; + var fileSearchTools = tools.Where(t => t is FileSearchTool).ToArray(); + Assert.Single(fileSearchTools); + FileSearchTool fileSearchTool = (fileSearchTools[0] as FileSearchTool)!; + Assert.NotNull(fileSearchTool); + + // Verify vector store content property exists and has correct values + Assert.NotNull(fileSearchTool.VectorStoreIds); + Assert.Equal(3, fileSearchTool.VectorStoreIds.LiteralValue.Length); + Assert.Equal("1", fileSearchTool.VectorStoreIds.LiteralValue[0]); + Assert.Equal("2", fileSearchTool.VectorStoreIds.LiteralValue[1]); + Assert.Equal("3", fileSearchTool.VectorStoreIds.LiteralValue[2]); + } + + [Fact] + public void FromYaml_ApiKeyConnection() + { + // Arrange & Act + var agent = AgentBotElementYaml.FromYaml(PromptAgents.AgentWithApiKeyConnection); + + // Assert + Assert.NotNull(agent); + Assert.NotNull(agent.Model); + CurrentModels model = (agent.Model as CurrentModels)!; + Assert.NotNull(model); + Assert.NotNull(model.Connection); + Assert.IsType(model.Connection); + ApiKeyConnection connection = (model.Connection as ApiKeyConnection)!; + Assert.NotNull(connection); + Assert.Equal("https://my-azure-openai-endpoint.openai.azure.com/", connection.Endpoint?.LiteralValue); + Assert.Equal("my-api-key", connection.Key?.LiteralValue); + } + + [Fact] + public void FromYaml_RemoteConnection() + { + // Arrange & Act + var agent = AgentBotElementYaml.FromYaml(PromptAgents.AgentWithRemoteConnection); + + // Assert + Assert.NotNull(agent); + Assert.NotNull(agent.Model); + CurrentModels model = (agent.Model as CurrentModels)!; + Assert.NotNull(model); + Assert.NotNull(model.Connection); + Assert.IsType(model.Connection); + RemoteConnection connection = (model.Connection as RemoteConnection)!; + Assert.NotNull(connection); + Assert.Equal("https://my-azure-openai-endpoint.openai.azure.com/", connection.Endpoint?.LiteralValue); + } + + [Fact] + public void FromYaml_WithVariableReferences() + { + // Arrange + IConfiguration configuration = new ConfigurationBuilder() + .AddInMemoryCollection(new Dictionary + { + ["OpenAIEndpoint"] = "endpoint", + ["OpenAIApiKey"] = "apiKey", + ["Temperature"] = "0.9", + ["TopP"] = "0.8" + }) + .Build(); + + // Act + var agent = AgentBotElementYaml.FromYaml(PromptAgents.AgentWithVariableReferences, configuration); + + // Assert + Assert.NotNull(agent); + Assert.NotNull(agent.Model); + CurrentModels model = (agent.Model as CurrentModels)!; + Assert.NotNull(model); + Assert.NotNull(model.Options); + Assert.Equal(0.9, Eval(model.Options?.Temperature, configuration)); + Assert.Equal(0.8, Eval(model.Options?.TopP, configuration)); + Assert.NotNull(model.Connection); + Assert.IsType(model.Connection); + ApiKeyConnection connection = (model.Connection as ApiKeyConnection)!; + Assert.NotNull(connection); + Assert.NotNull(connection.Endpoint); + Assert.NotNull(connection.Key); + Assert.Equal("endpoint", Eval(connection.Endpoint, configuration)); + Assert.Equal("apiKey", Eval(connection.Key, configuration)); + } + + /// + /// Represents information about a person, including their name, age, and occupation, matched to the JSON schema used in the agent. + /// + [Description("Information about a person including their name, age, and occupation")] + public sealed class PersonInfo + { + [JsonPropertyName("name")] + public string? Name { get; set; } + + [JsonPropertyName("age")] + public int? Age { get; set; } + + [JsonPropertyName("occupation")] + public string? Occupation { get; set; } + } + + private static string? Eval(StringExpression? expression, IConfiguration? configuration = null) + { + if (expression is null) + { + return null; + } + + RecalcEngine engine = new(); + if (configuration is not null) + { + foreach (var kvp in configuration.AsEnumerable()) + { + engine.UpdateVariable(kvp.Key, kvp.Value ?? string.Empty); + } + } + + return expression.Eval(engine); + } + + private static double? Eval(NumberExpression? expression, IConfiguration? configuration = null) + { + if (expression is null) + { + return null; + } + + RecalcEngine engine = new(); + if (configuration != null) + { + foreach (var kvp in configuration.AsEnumerable()) + { + engine.UpdateVariable(kvp.Key, kvp.Value ?? string.Empty); + } + } + + return expression.Eval(engine); + } +} diff --git a/dotnet/tests/Microsoft.Agents.AI.Declarative.UnitTests/AggregatorPromptAgentFactoryTests.cs b/dotnet/tests/Microsoft.Agents.AI.Declarative.UnitTests/AggregatorPromptAgentFactoryTests.cs new file mode 100644 index 0000000000..d20bd9be00 --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.Declarative.UnitTests/AggregatorPromptAgentFactoryTests.cs @@ -0,0 +1,89 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Collections.Generic; +using System.Text.Json; +using System.Threading; +using System.Threading.Tasks; +using Microsoft.Bot.ObjectModel; +using Microsoft.Extensions.AI; + +namespace Microsoft.Agents.AI.Declarative.UnitTests; + +/// +/// Unit tests for +/// +public sealed class AggregatorPromptAgentFactoryTests +{ + [Fact] + public void AggregatorAgentFactory_ThrowsForEmptyArray() + { + // Arrange & Act & Assert + Assert.Throws(() => new AggregatorPromptAgentFactory([])); + } + + [Fact] + public async Task AggregatorAgentFactory_ReturnsNull() + { + // Arrange + var factory = new AggregatorPromptAgentFactory([new TestAgentFactory(null)]); + + // Act + var agent = await factory.TryCreateAsync(new GptComponentMetadata("test")); + + // Assert + Assert.Null(agent); + } + + [Fact] + public async Task AggregatorAgentFactory_ReturnsAgent() + { + // Arrange + var agentToReturn = new TestAgent(); + var factory = new AggregatorPromptAgentFactory([new TestAgentFactory(null), new TestAgentFactory(agentToReturn)]); + + // Act + var agent = await factory.TryCreateAsync(new GptComponentMetadata("test")); + + // Assert + Assert.Equal(agentToReturn, agent); + } + + private sealed class TestAgentFactory : PromptAgentFactory + { + private readonly AIAgent? _agentToReturn; + + public TestAgentFactory(AIAgent? agentToReturn = null) + { + this._agentToReturn = agentToReturn; + } + + public override Task TryCreateAsync(GptComponentMetadata promptAgent, CancellationToken cancellationToken = default) + { + return Task.FromResult(this._agentToReturn); + } + } + + private sealed class TestAgent : AIAgent + { + public override AgentThread DeserializeThread(JsonElement serializedThread, JsonSerializerOptions? jsonSerializerOptions = null) + { + throw new NotImplementedException(); + } + + public override AgentThread GetNewThread() + { + throw new NotImplementedException(); + } + + public override Task RunAsync(IEnumerable messages, AgentThread? thread = null, AgentRunOptions? options = null, CancellationToken cancellationToken = default) + { + throw new NotImplementedException(); + } + + public override IAsyncEnumerable RunStreamingAsync(IEnumerable messages, AgentThread? thread = null, AgentRunOptions? options = null, CancellationToken cancellationToken = default) + { + throw new NotImplementedException(); + } + } +} diff --git a/dotnet/tests/Microsoft.Agents.AI.Declarative.UnitTests/ChatClient/ChatClientAgentFactoryTests.cs b/dotnet/tests/Microsoft.Agents.AI.Declarative.UnitTests/ChatClient/ChatClientAgentFactoryTests.cs new file mode 100644 index 0000000000..8590662000 --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.Declarative.UnitTests/ChatClient/ChatClientAgentFactoryTests.cs @@ -0,0 +1,107 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Threading.Tasks; +using Microsoft.Extensions.AI; +using Moq; + +namespace Microsoft.Agents.AI.Declarative.UnitTests.ChatClient; + +/// +/// Unit tests for . +/// +public sealed class ChatClientAgentFactoryTests +{ + private readonly Mock _mockChatClient; + + public ChatClientAgentFactoryTests() + { + this._mockChatClient = new(); + } + + [Fact] + public async Task TryCreateAsync_WithChatClientInConstructor_CreatesAgentAsync() + { + // Arrange + var promptAgent = PromptAgents.CreateTestPromptAgent(); + ChatClientPromptAgentFactory factory = new(this._mockChatClient.Object); + + // Act + AIAgent? agent = await factory.TryCreateAsync(promptAgent); + + // Assert + Assert.NotNull(agent); + Assert.IsType(agent); + Assert.Equal("Test Agent", agent.Name); + Assert.Equal("Test Description", agent.Description); + } + + [Fact] + public async Task TryCreateAsync_Creates_ChatClientAgentAsync() + { + // Arrange + var promptAgent = PromptAgents.CreateTestPromptAgent(); + ChatClientPromptAgentFactory factory = new(this._mockChatClient.Object); + + // Act + AIAgent? agent = await factory.TryCreateAsync(promptAgent); + + // Assert + Assert.NotNull(agent); + Assert.IsType(agent); + var chatClientAgent = agent as ChatClientAgent; + Assert.NotNull(chatClientAgent); + Assert.Equal("You are a helpful assistant.", chatClientAgent.Instructions); + Assert.NotNull(chatClientAgent.ChatClient); + Assert.NotNull(chatClientAgent.ChatOptions); + } + + [Fact] + public async Task TryCreateAsync_Creates_ChatOptionsAsync() + { + // Arrange + var promptAgent = PromptAgents.CreateTestPromptAgent(); + ChatClientPromptAgentFactory factory = new(this._mockChatClient.Object); + + // Act + AIAgent? agent = await factory.TryCreateAsync(promptAgent); + + // Assert + Assert.NotNull(agent); + Assert.IsType(agent); + var chatClientAgent = agent as ChatClientAgent; + Assert.NotNull(chatClientAgent?.ChatOptions); + Assert.Equal("You are a helpful assistant.", chatClientAgent?.ChatOptions?.Instructions); + Assert.Equal(0.7F, chatClientAgent?.ChatOptions?.Temperature); + Assert.Equal(0.7F, chatClientAgent?.ChatOptions?.FrequencyPenalty); + Assert.Equal(1024, chatClientAgent?.ChatOptions?.MaxOutputTokens); + Assert.Equal(0.9F, chatClientAgent?.ChatOptions?.TopP); + Assert.Equal(50, chatClientAgent?.ChatOptions?.TopK); + Assert.Equal(0.7F, chatClientAgent?.ChatOptions?.PresencePenalty); + Assert.Equal(42L, chatClientAgent?.ChatOptions?.Seed); + Assert.NotNull(chatClientAgent?.ChatOptions?.ResponseFormat); + Assert.Equal("gpt-4o", chatClientAgent?.ChatOptions?.ModelId); + Assert.Equal(["###", "END", "STOP"], chatClientAgent?.ChatOptions?.StopSequences); + Assert.True(chatClientAgent?.ChatOptions?.AllowMultipleToolCalls); + Assert.Equal(ChatToolMode.Auto, chatClientAgent?.ChatOptions?.ToolMode); + Assert.Equal("customValue", chatClientAgent?.ChatOptions?.AdditionalProperties?["customProperty"]); + } + + [Fact] + public async Task TryCreateAsync_Creates_ToolsAsync() + { + // Arrange + var promptAgent = PromptAgents.CreateTestPromptAgent(); + ChatClientPromptAgentFactory factory = new(this._mockChatClient.Object); + + // Act + AIAgent? agent = await factory.TryCreateAsync(promptAgent); + + // Assert + Assert.NotNull(agent); + Assert.IsType(agent); + var chatClientAgent = agent as ChatClientAgent; + Assert.NotNull(chatClientAgent?.ChatOptions?.Tools); + var tools = chatClientAgent?.ChatOptions?.Tools; + Assert.Equal(5, tools?.Count); + } +} diff --git a/dotnet/tests/Microsoft.Agents.AI.Declarative.UnitTests/Microsoft.Agents.AI.Declarative.UnitTests.csproj b/dotnet/tests/Microsoft.Agents.AI.Declarative.UnitTests/Microsoft.Agents.AI.Declarative.UnitTests.csproj new file mode 100644 index 0000000000..d348a0b433 --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.Declarative.UnitTests/Microsoft.Agents.AI.Declarative.UnitTests.csproj @@ -0,0 +1,17 @@ + + + + $(NoWarn);IDE1006;VSTHRD200 + + + + + + + + + + + + + diff --git a/dotnet/tests/Microsoft.Agents.AI.Declarative.UnitTests/PromptAgents.cs b/dotnet/tests/Microsoft.Agents.AI.Declarative.UnitTests/PromptAgents.cs new file mode 100644 index 0000000000..01fa2026fb --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.Declarative.UnitTests/PromptAgents.cs @@ -0,0 +1,386 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Microsoft.Bot.ObjectModel; + +namespace Microsoft.Agents.AI.Declarative.UnitTests; + +internal static class PromptAgents +{ + internal const string AgentWithEverything = + """ + kind: Prompt + name: AgentName + description: Agent description + instructions: You are a helpful assistant. + model: + id: gpt-4o + options: + temperature: 0.7 + maxOutputTokens: 1024 + topP: 0.9 + topK: 50 + frequencyPenalty: 0.0 + presencePenalty: 0.0 + seed: 42 + responseFormat: text + stopSequences: + - "###" + - "END" + - "STOP" + allowMultipleToolCalls: true + tools: + - kind: codeInterpreter + inputs: + - kind: HostedFileContent + FileId: fileId123 + - kind: function + name: GetWeather + description: Get the weather for a given location. + parameters: + - name: location + type: string + description: The city and state, e.g. San Francisco, CA + required: true + - name: unit + type: string + description: The unit of temperature. Possible values are 'celsius' and 'fahrenheit'. + required: false + enum: + - celsius + - fahrenheit + - kind: mcp + serverName: PersonInfoTool + serverDescription: Get information about a person. + connection: + kind: AnonymousConnection + endpoint: https://my-mcp-endpoint.com/api + allowedTools: + - "GetPersonInfo" + - "UpdatePersonInfo" + - "DeletePersonInfo" + approvalMode: + kind: HostedMcpServerToolRequireSpecificApprovalMode + AlwaysRequireApprovalToolNames: + - "UpdatePersonInfo" + - "DeletePersonInfo" + NeverRequireApprovalToolNames: + - "GetPersonInfo" + - kind: webSearch + name: WebSearchTool + description: Search the web for information. + - kind: fileSearch + name: FileSearchTool + description: Search files for information. + ranker: default + scoreThreshold: 0.5 + maxResults: 5 + maxContentLength: 2000 + vectorStoreIds: + - 1 + - 2 + - 3 + """; + + internal const string AgentWithOutputSchema = + """ + kind: Prompt + name: Translation Assistant + description: A helpful assistant that translates text to a specified language. + model: + id: gpt-4o + options: + temperature: 0.9 + topP: 0.95 + instructions: You are a helpful assistant. You answer questions in {language}. You return your answers in a JSON format. + additionalInstructions: You must always respond in the specified language. + tools: + - kind: codeInterpreter + template: + format: PowerFx # Mustache is the other option + parser: None # Prompty and XML are the other options + inputSchema: + properties: + language: string + outputSchema: + properties: + language: + type: string + required: true + description: The language of the answer. + answer: + type: string + required: true + description: The answer text. + """; + + internal const string AgentWithApiKeyConnection = + """ + kind: Prompt + name: AgentName + description: Agent description + instructions: You are a helpful assistant. + model: + id: gpt-4o + connection: + kind: ApiKey + endpoint: https://my-azure-openai-endpoint.openai.azure.com/ + key: my-api-key + """; + + internal const string AgentWithRemoteConnection = + """ + kind: Prompt + name: AgentName + description: Agent description + instructions: You are a helpful assistant. + model: + id: gpt-4o + connection: + kind: Remote + endpoint: https://my-azure-openai-endpoint.openai.azure.com/ + """; + + internal const string AgentWithVariableReferences = + """ + kind: Prompt + name: AgentName + description: Agent description + instructions: You are a helpful assistant. + model: + id: gpt-4o + options: + temperature: =Env.Temperature + topP: =Env.TopP + connection: + kind: apiKey + endpoint: =Env.OpenAIEndpoint + key: =Env.OpenAIApiKey + """; + + internal const string OpenAIChatAgent = + """ + kind: Prompt + name: Assistant + description: Helpful assistant + instructions: You are a helpful assistant. You answer questions in the language specified by the user. You return your answers in a JSON format. + model: + id: =Env.OPENAI_MODEL + options: + temperature: 0.9 + topP: 0.95 + connection: + kind: apiKey + key: =Env.OPENAI_API_KEY + outputSchema: + properties: + language: + type: string + required: true + description: The language of the answer. + answer: + type: string + required: true + description: The answer text. + """; + + internal const string AgentWithCurrentModels = + """ + kind: Prompt + name: AgentName + description: Agent description + instructions: You are a helpful assistant. + model: + id: gpt-4o + options: + temperature: 0.7 + maxOutputTokens: 1024 + topP: 0.9 + topK: 50 + frequencyPenalty: 0.7 + presencePenalty: 0.7 + seed: 42 + responseFormat: text + stopSequences: + - "###" + - "END" + - "STOP" + allowMultipleToolCalls: true + chatToolMode: auto + """; + + internal const string AgentWithCurrentModelsSnakeCase = + """ + kind: Prompt + name: AgentName + description: Agent description + instructions: You are a helpful assistant. + model: + id: gpt-4o + options: + temperature: 0.7 + max_output_tokens: 1024 + top_p: 0.9 + top_k: 50 + frequency_penalty: 0.7 + presence_penalty: 0.7 + seed: 42 + response_format: text + stop_sequences: + - "###" + - "END" + - "STOP" + allow_multiple_tool_calls: true + chat_tool_mode: auto + """; + + internal const string Workflow = + """ + kind: Workflow + trigger: + + kind: OnConversationStart + id: workflow_demo + actions: + + - kind: InvokeAzureAgent + id: question_student + conversationId: =System.ConversationId + agent: + name: StudentAgent + + - kind: InvokeAzureAgent + id: question_teacher + conversationId: =System.ConversationId + agent: + name: TeacherAgent + output: + messages: Local.TeacherResponse + + - kind: SetVariable + id: set_count_increment + variable: Local.TurnCount + value: =Local.TurnCount + 1 + + - kind: ConditionGroup + id: check_completion + conditions: + + - condition: =!IsBlank(Find("CONGRATULATIONS", Upper(MessageText(Local.TeacherResponse)))) + id: check_turn_done + actions: + + - kind: SendActivity + id: sendActivity_done + activity: GOLD STAR! + + - condition: =Local.TurnCount < 4 + id: check_turn_count + actions: + + - kind: GotoAction + id: goto_student_agent + actionId: question_student + + elseActions: + + - kind: SendActivity + id: sendActivity_tired + activity: Let's try again later... + + """; + + internal static readonly string[] s_stopSequences = ["###", "END", "STOP"]; + + internal static GptComponentMetadata CreateTestPromptAgent(string? publisher = "OpenAI", string? apiType = "Chat") + { + string agentYaml = + $""" + kind: Prompt + name: Test Agent + description: Test Description + instructions: You are a helpful assistant. + additionalInstructions: Provide detailed and accurate responses. + model: + id: gpt-4o + publisher: {publisher} + apiType: {apiType} + options: + modelId: gpt-4o + temperature: 0.7 + maxOutputTokens: 1024 + topP: 0.9 + topK: 50 + frequencyPenalty: 0.7 + presencePenalty: 0.7 + seed: 42 + responseFormat: text + stopSequences: + - "###" + - "END" + - "STOP" + allowMultipleToolCalls: true + chatToolMode: auto + customProperty: customValue + connection: + kind: apiKey + endpoint: https://my-azure-openai-endpoint.openai.azure.com/ + key: my-api-key + tools: + - kind: codeInterpreter + - kind: function + name: GetWeather + description: Get the weather for a given location. + parameters: + - name: location + type: string + description: The city and state, e.g. San Francisco, CA + required: true + - name: unit + type: string + description: The unit of temperature. Possible values are 'celsius' and 'fahrenheit'. + required: false + enum: + - celsius + - fahrenheit + - kind: mcp + serverName: PersonInfoTool + serverDescription: Get information about a person. + allowedTools: + - "GetPersonInfo" + - "UpdatePersonInfo" + - "DeletePersonInfo" + approvalMode: + kind: HostedMcpServerToolRequireSpecificApprovalMode + AlwaysRequireApprovalToolNames: + - "UpdatePersonInfo" + - "DeletePersonInfo" + NeverRequireApprovalToolNames: + - "GetPersonInfo" + connection: + kind: AnonymousConnection + endpoint: https://my-mcp-endpoint.com/api + - kind: webSearch + name: WebSearchTool + description: Search the web for information. + - kind: fileSearch + name: FileSearchTool + description: Search files for information. + vectorStoreIds: + - 1 + - 2 + - 3 + outputSchema: + properties: + language: + type: string + required: true + description: The language of the answer. + answer: + type: string + required: true + description: The answer text. + """; + + return AgentBotElementYaml.FromYaml(agentYaml); + } +} diff --git a/dotnet/tests/Microsoft.Agents.AI.DevUI.UnitTests/DevUIExtensionsTests.cs b/dotnet/tests/Microsoft.Agents.AI.DevUI.UnitTests/DevUIExtensionsTests.cs new file mode 100644 index 0000000000..d002068626 --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.DevUI.UnitTests/DevUIExtensionsTests.cs @@ -0,0 +1,222 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using Microsoft.Agents.AI.Workflows; +using Microsoft.Extensions.AI; +using Microsoft.Extensions.DependencyInjection; +using Moq; + +namespace Microsoft.Agents.AI.DevUI.UnitTests; + +/// +/// Unit tests for DevUI service collection extensions. +/// Tests verify that workflows and agents can be resolved even when registered non-conventionally. +/// +public class DevUIExtensionsTests +{ + /// + /// Verifies that AddDevUI throws ArgumentNullException when services collection is null. + /// + [Fact] + public void AddDevUI_NullServices_ThrowsArgumentNullException() + { + IServiceCollection services = null!; + Assert.Throws(() => services.AddDevUI()); + } + + /// + /// Verifies that GetRequiredKeyedService throws for non-existent keys. + /// + [Fact] + public void AddDevUI_GetRequiredKeyedServiceNonExistent_ThrowsInvalidOperationException() + { + // Arrange + var services = new ServiceCollection(); + services.AddDevUI(); + var serviceProvider = services.BuildServiceProvider(); + + // Act & Assert + Assert.Throws(() => serviceProvider.GetRequiredKeyedService("non-existent")); + } + + /// + /// Verifies that an agent with null name can be resolved by its workflow. + /// + [Fact] + public void AddDevUI_WorkflowWithName_CanBeResolved_AsAIAgent() + { + // Arrange + var services = new ServiceCollection(); + var mockChatClient = new Mock(); + var agent1 = new ChatClientAgent(mockChatClient.Object, "Test 1", name: null); + var agent2 = new ChatClientAgent(mockChatClient.Object, "Test 2", name: null); + var workflow = AgentWorkflowBuilder.BuildSequential(agent1, agent2); + + services.AddKeyedSingleton("workflow", workflow); + services.AddDevUI(); + + var serviceProvider = services.BuildServiceProvider(); + + // Act + var resolvedWorkflowAsAgent = serviceProvider.GetKeyedService("workflow"); + + // Assert + Assert.NotNull(resolvedWorkflowAsAgent); + Assert.Null(resolvedWorkflowAsAgent.Name); + } + + /// + /// Verifies that an agent with null name can be resolved by its workflow. + /// + [Fact] + public void AddDevUI_MultipleWorkflowsWithName_CanBeResolved_AsAIAgent() + { + var services = new ServiceCollection(); + var mockChatClient = new Mock(); + var agent1 = new ChatClientAgent(mockChatClient.Object, "Test 1", name: null); + var agent2 = new ChatClientAgent(mockChatClient.Object, "Test 2", name: null); + var workflow1 = AgentWorkflowBuilder.BuildSequential(agent1, agent2); + var workflow2 = AgentWorkflowBuilder.BuildSequential(agent1, agent2); + + services.AddKeyedSingleton("workflow1", workflow1); + services.AddKeyedSingleton("workflow2", workflow2); + services.AddDevUI(); + + var serviceProvider = services.BuildServiceProvider(); + + var resolvedWorkflow1AsAgent = serviceProvider.GetKeyedService("workflow1"); + Assert.NotNull(resolvedWorkflow1AsAgent); + Assert.Null(resolvedWorkflow1AsAgent.Name); + + var resolvedWorkflow2AsAgent = serviceProvider.GetKeyedService("workflow2"); + Assert.NotNull(resolvedWorkflow2AsAgent); + Assert.Null(resolvedWorkflow2AsAgent.Name); + + Assert.False(resolvedWorkflow1AsAgent == resolvedWorkflow2AsAgent); + } + + /// + /// Verifies that an agent with null name can be resolved by its workflow. + /// + [Fact] + public void AddDevUI_NonKeyedWorkflow_CanBeResolved_AsAIAgent() + { + var services = new ServiceCollection(); + var mockChatClient = new Mock(); + var agent1 = new ChatClientAgent(mockChatClient.Object, "Test 1", name: null); + var agent2 = new ChatClientAgent(mockChatClient.Object, "Test 2", name: null); + var workflow = AgentWorkflowBuilder.BuildSequential(agent1, agent2); + + services.AddKeyedSingleton("workflow", workflow); + services.AddDevUI(); + + var serviceProvider = services.BuildServiceProvider(); + + var resolvedWorkflowAsAgent = serviceProvider.GetKeyedService("workflow"); + Assert.NotNull(resolvedWorkflowAsAgent); + Assert.Null(resolvedWorkflowAsAgent.Name); + } + + /// + /// Verifies that an agent with null name can be resolved by its workflow. + /// + [Fact] + public void AddDevUI_NonKeyedWorkflow_PlusKeyedWorkflow_CanBeResolved_AsAIAgent() + { + var services = new ServiceCollection(); + var mockChatClient = new Mock(); + var agent1 = new ChatClientAgent(mockChatClient.Object, "Test 1", name: null); + var agent2 = new ChatClientAgent(mockChatClient.Object, "Test 2", name: null); + var workflow = AgentWorkflowBuilder.BuildSequential("standardname", agent1, agent2); + var keyedWorkflow = AgentWorkflowBuilder.BuildSequential("keyedname", agent1, agent2); + + services.AddSingleton(workflow); + services.AddKeyedSingleton("keyed", keyedWorkflow); + services.AddDevUI(); + + var serviceProvider = services.BuildServiceProvider(); + + // resolve a workflow with the same name as workflow's name (which is registered without a key) + var standardAgent = serviceProvider.GetKeyedService("standardname"); + Assert.NotNull(standardAgent); + Assert.Equal("standardname", standardAgent.Name); + + var keyedAgent = serviceProvider.GetKeyedService("keyed"); + Assert.NotNull(keyedAgent); + Assert.Equal("keyedname", keyedAgent.Name); + + var nonExisting = serviceProvider.GetKeyedService("random-non-existing!!!"); + Assert.Null(nonExisting); + } + + /// + /// Verifies that an agent registered with a different key than its name can be resolved by key. + /// + [Fact] + public void AddDevUI_AgentRegisteredWithDifferentKey_CanBeResolvedByKey() + { + // Arrange + var services = new ServiceCollection(); + const string AgentName = "actual-agent-name"; + const string RegistrationKey = "different-key"; + var mockChatClient = new Mock(); + var agent = new ChatClientAgent(mockChatClient.Object, "Test", AgentName); + + services.AddKeyedSingleton(RegistrationKey, agent); + services.AddDevUI(); + + var serviceProvider = services.BuildServiceProvider(); + + // Act + var resolvedAgent = serviceProvider.GetKeyedService(RegistrationKey); + + // Assert + Assert.NotNull(resolvedAgent); + // The resolved agent should have the agent's name, not the registration key + Assert.Equal(AgentName, resolvedAgent.Name); + } + + /// + /// Verifies that an agent registered with a different key than its name can be resolved by key. + /// + [Fact] + public void AddDevUI_Keyed_AndStandard_BothCanBeResolved() + { + // Arrange + var services = new ServiceCollection(); + var mockChatClient = new Mock(); + var defaultAgent = new ChatClientAgent(mockChatClient.Object, "default", "default"); + var keyedAgent = new ChatClientAgent(mockChatClient.Object, "keyed", "keyed"); + + services.AddSingleton(defaultAgent); + services.AddKeyedSingleton("keyed-registration", keyedAgent); + services.AddDevUI(); + + var serviceProvider = services.BuildServiceProvider(); + + var resolvedKeyedAgent = serviceProvider.GetKeyedService("keyed-registration"); + Assert.NotNull(resolvedKeyedAgent); + Assert.Equal("keyed", resolvedKeyedAgent.Name); + + // resolving default agent based on its name, not on the registration-key + var resolvedDefaultAgent = serviceProvider.GetKeyedService("default"); + Assert.NotNull(resolvedDefaultAgent); + Assert.Equal("default", resolvedDefaultAgent.Name); + } + + /// + /// Verifies that the DevUI fallback handler error message includes helpful information. + /// + [Fact] + public void AddDevUI_InvalidResolution_ErrorMessageIsInformative() + { + // Arrange + var services = new ServiceCollection(); + services.AddDevUI(); + var serviceProvider = services.BuildServiceProvider(); + const string InvalidKey = "invalid-key-name"; + + // Act & Assert + var exception = Assert.Throws(() => serviceProvider.GetRequiredKeyedService(InvalidKey)); + } +} diff --git a/dotnet/tests/Microsoft.Agents.AI.DevUI.UnitTests/DevUIIntegrationTests.cs b/dotnet/tests/Microsoft.Agents.AI.DevUI.UnitTests/DevUIIntegrationTests.cs new file mode 100644 index 0000000000..b8512a856e --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.DevUI.UnitTests/DevUIIntegrationTests.cs @@ -0,0 +1,285 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Net.Http.Json; +using System.Threading.Tasks; +using Microsoft.Agents.AI.DevUI.Entities; +using Microsoft.Agents.AI.Workflows; +using Microsoft.AspNetCore.Builder; +using Microsoft.AspNetCore.TestHost; +using Microsoft.Extensions.AI; +using Microsoft.Extensions.DependencyInjection; +using Moq; + +namespace Microsoft.Agents.AI.DevUI.UnitTests; + +public class DevUIIntegrationTests +{ + private sealed class NoOpExecutor(string id) : Executor(id) + { + protected override RouteBuilder ConfigureRoutes(RouteBuilder routeBuilder) => + routeBuilder.AddHandler( + (msg, ctx) => ctx.SendMessageAsync(msg)); + } + + [Fact] + public async Task TestServerWithDevUI_ResolvesRequestToWorkflow_ByKeyAsync() + { + // Arrange + WebApplicationBuilder builder = WebApplication.CreateBuilder(); + builder.WebHost.UseTestServer(); + + var mockChatClient = new Mock(); + var agent = new ChatClientAgent(mockChatClient.Object, "Test", "agent-name"); + + builder.Services.AddKeyedSingleton("registration-key", agent); + builder.Services.AddDevUI(); + + using WebApplication app = builder.Build(); + app.MapDevUI(); + + await app.StartAsync(); + + // Act + var resolvedAgent = app.Services.GetKeyedService("registration-key"); + var client = app.GetTestClient(); + var response = await client.GetAsync(new Uri("/v1/entities", uriKind: UriKind.Relative)); + + var discoveryResponse = await response.Content.ReadFromJsonAsync(); + Assert.NotNull(discoveryResponse); + Assert.Single(discoveryResponse.Entities); + Assert.Equal("agent-name", discoveryResponse.Entities[0].Name); + } + + [Fact] + public async Task TestServerWithDevUI_ResolvesMultipleAIAgents_ByKeyAsync() + { + // Arrange + WebApplicationBuilder builder = WebApplication.CreateBuilder(); + builder.WebHost.UseTestServer(); + + var mockChatClient = new Mock(); + var agent1 = new ChatClientAgent(mockChatClient.Object, "Test", "agent-one"); + var agent2 = new ChatClientAgent(mockChatClient.Object, "Test", "agent-two"); + var agent3 = new ChatClientAgent(mockChatClient.Object, "Test", "agent-three"); + + builder.Services.AddKeyedSingleton("key-1", agent1); + builder.Services.AddKeyedSingleton("key-2", agent2); + builder.Services.AddKeyedSingleton("key-3", agent3); + builder.Services.AddDevUI(); + + using WebApplication app = builder.Build(); + app.MapDevUI(); + + await app.StartAsync(); + + // Act + var client = app.GetTestClient(); + var response = await client.GetAsync(new Uri("/v1/entities", uriKind: UriKind.Relative)); + + var discoveryResponse = await response.Content.ReadFromJsonAsync(); + + // Assert + Assert.NotNull(discoveryResponse); + Assert.Equal(3, discoveryResponse.Entities.Count); + Assert.Contains(discoveryResponse.Entities, e => e.Name == "agent-one" && e.Type == "agent"); + Assert.Contains(discoveryResponse.Entities, e => e.Name == "agent-two" && e.Type == "agent"); + Assert.Contains(discoveryResponse.Entities, e => e.Name == "agent-three" && e.Type == "agent"); + } + + [Fact] + public async Task TestServerWithDevUI_ResolvesAIAgents_WithKeyedAndDefaultRegistrationAsync() + { + // Arrange + WebApplicationBuilder builder = WebApplication.CreateBuilder(); + builder.WebHost.UseTestServer(); + + var mockChatClient = new Mock(); + var agentKeyed1 = new ChatClientAgent(mockChatClient.Object, "Test", "keyed-agent-one"); + var agentKeyed2 = new ChatClientAgent(mockChatClient.Object, "Test", "keyed-agent-two"); + var agentDefault = new ChatClientAgent(mockChatClient.Object, "Test", "default-agent"); + + builder.Services.AddKeyedSingleton("key-1", agentKeyed1); + builder.Services.AddKeyedSingleton("key-2", agentKeyed2); + builder.Services.AddSingleton(agentDefault); + builder.Services.AddDevUI(); + + using WebApplication app = builder.Build(); + app.MapDevUI(); + + await app.StartAsync(); + + // Act + var client = app.GetTestClient(); + var response = await client.GetAsync(new Uri("/v1/entities", uriKind: UriKind.Relative)); + + var discoveryResponse = await response.Content.ReadFromJsonAsync(); + + // Assert + Assert.NotNull(discoveryResponse); + Assert.Equal(3, discoveryResponse.Entities.Count); + Assert.Contains(discoveryResponse.Entities, e => e.Name == "keyed-agent-one" && e.Type == "agent"); + Assert.Contains(discoveryResponse.Entities, e => e.Name == "keyed-agent-two" && e.Type == "agent"); + Assert.Contains(discoveryResponse.Entities, e => e.Name == "default-agent" && e.Type == "agent"); + } + + [Fact] + public async Task TestServerWithDevUI_ResolvesMultipleWorkflows_ByKeyAsync() + { + // Arrange + WebApplicationBuilder builder = WebApplication.CreateBuilder(); + builder.WebHost.UseTestServer(); + + var workflow1 = new WorkflowBuilder("executor-1") + .WithName("workflow-one") + .WithDescription("First workflow") + .BindExecutor(new NoOpExecutor("executor-1")) + .Build(); + + var workflow2 = new WorkflowBuilder("executor-2") + .WithName("workflow-two") + .WithDescription("Second workflow") + .BindExecutor(new NoOpExecutor("executor-2")) + .Build(); + + var workflow3 = new WorkflowBuilder("executor-3") + .WithName("workflow-three") + .WithDescription("Third workflow") + .BindExecutor(new NoOpExecutor("executor-3")) + .Build(); + + builder.Services.AddKeyedSingleton("key-1", workflow1); + builder.Services.AddKeyedSingleton("key-2", workflow2); + builder.Services.AddKeyedSingleton("key-3", workflow3); + builder.Services.AddDevUI(); + + using WebApplication app = builder.Build(); + app.MapDevUI(); + + await app.StartAsync(); + + // Act + var client = app.GetTestClient(); + var response = await client.GetAsync(new Uri("/v1/entities", uriKind: UriKind.Relative)); + + var discoveryResponse = await response.Content.ReadFromJsonAsync(); + + // Assert + Assert.NotNull(discoveryResponse); + Assert.Equal(3, discoveryResponse.Entities.Count); + Assert.Contains(discoveryResponse.Entities, e => e.Name == "workflow-one" && e.Type == "workflow"); + Assert.Contains(discoveryResponse.Entities, e => e.Name == "workflow-two" && e.Type == "workflow"); + Assert.Contains(discoveryResponse.Entities, e => e.Name == "workflow-three" && e.Type == "workflow"); + } + + [Fact] + public async Task TestServerWithDevUI_ResolvesWorkflows_WithKeyedAndDefaultRegistrationAsync() + { + // Arrange + WebApplicationBuilder builder = WebApplication.CreateBuilder(); + builder.WebHost.UseTestServer(); + + var workflowKeyed1 = new WorkflowBuilder("executor-1") + .WithName("keyed-workflow-one") + .BindExecutor(new NoOpExecutor("executor-1")) + .Build(); + + var workflowKeyed2 = new WorkflowBuilder("executor-2") + .WithName("keyed-workflow-two") + .BindExecutor(new NoOpExecutor("executor-2")) + .Build(); + + var workflowDefault = new WorkflowBuilder("executor-default") + .WithName("default-workflow") + .BindExecutor(new NoOpExecutor("executor-default")) + .Build(); + + builder.Services.AddKeyedSingleton("key-1", workflowKeyed1); + builder.Services.AddKeyedSingleton("key-2", workflowKeyed2); + builder.Services.AddSingleton(workflowDefault); + builder.Services.AddDevUI(); + + using WebApplication app = builder.Build(); + app.MapDevUI(); + + await app.StartAsync(); + + // Act + var client = app.GetTestClient(); + var response = await client.GetAsync(new Uri("/v1/entities", uriKind: UriKind.Relative)); + + var discoveryResponse = await response.Content.ReadFromJsonAsync(); + + // Assert + Assert.NotNull(discoveryResponse); + Assert.Equal(3, discoveryResponse.Entities.Count); + Assert.Contains(discoveryResponse.Entities, e => e.Name == "keyed-workflow-one" && e.Type == "workflow"); + Assert.Contains(discoveryResponse.Entities, e => e.Name == "keyed-workflow-two" && e.Type == "workflow"); + Assert.Contains(discoveryResponse.Entities, e => e.Name == "default-workflow" && e.Type == "workflow"); + } + + [Fact] + public async Task TestServerWithDevUI_ResolvesMixedAgentsAndWorkflows_AllRegistrationsAsync() + { + // Arrange + WebApplicationBuilder builder = WebApplication.CreateBuilder(); + builder.WebHost.UseTestServer(); + + var mockChatClient = new Mock(); + + // Create AIAgents + var agent1 = new ChatClientAgent(mockChatClient.Object, "Test", "mixed-agent-one"); + var agent2 = new ChatClientAgent(mockChatClient.Object, "Test", "mixed-agent-two"); + var agentDefault = new ChatClientAgent(mockChatClient.Object, "Test", "default-mixed-agent"); + + // Create Workflows + var workflow1 = new WorkflowBuilder("executor-1") + .WithName("mixed-workflow-one") + .BindExecutor(new NoOpExecutor("executor-1")) + .Build(); + + var workflow2 = new WorkflowBuilder("executor-2") + .WithName("mixed-workflow-two") + .BindExecutor(new NoOpExecutor("executor-2")) + .Build(); + + var workflowDefault = new WorkflowBuilder("executor-default") + .WithName("default-mixed-workflow") + .BindExecutor(new NoOpExecutor("executor-default")) + .Build(); + + // Register all + builder.Services.AddKeyedSingleton("agent-key-1", agent1); + builder.Services.AddKeyedSingleton("agent-key-2", agent2); + builder.Services.AddSingleton(agentDefault); + builder.Services.AddKeyedSingleton("workflow-key-1", workflow1); + builder.Services.AddKeyedSingleton("workflow-key-2", workflow2); + builder.Services.AddSingleton(workflowDefault); + builder.Services.AddDevUI(); + + using WebApplication app = builder.Build(); + app.MapDevUI(); + + await app.StartAsync(); + + // Act + var client = app.GetTestClient(); + var response = await client.GetAsync(new Uri("/v1/entities", uriKind: UriKind.Relative)); + + var discoveryResponse = await response.Content.ReadFromJsonAsync(); + + // Assert + Assert.NotNull(discoveryResponse); + Assert.Equal(6, discoveryResponse.Entities.Count); + + // Verify agents + Assert.Contains(discoveryResponse.Entities, e => e.Name == "mixed-agent-one" && e.Type == "agent"); + Assert.Contains(discoveryResponse.Entities, e => e.Name == "mixed-agent-two" && e.Type == "agent"); + Assert.Contains(discoveryResponse.Entities, e => e.Name == "default-mixed-agent" && e.Type == "agent"); + + // Verify workflows + Assert.Contains(discoveryResponse.Entities, e => e.Name == "mixed-workflow-one" && e.Type == "workflow"); + Assert.Contains(discoveryResponse.Entities, e => e.Name == "mixed-workflow-two" && e.Type == "workflow"); + Assert.Contains(discoveryResponse.Entities, e => e.Name == "default-mixed-workflow" && e.Type == "workflow"); + } +} diff --git a/dotnet/tests/Microsoft.Agents.AI.DevUI.UnitTests/Microsoft.Agents.AI.DevUI.UnitTests.csproj b/dotnet/tests/Microsoft.Agents.AI.DevUI.UnitTests/Microsoft.Agents.AI.DevUI.UnitTests.csproj new file mode 100644 index 0000000000..1fc964e702 --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.DevUI.UnitTests/Microsoft.Agents.AI.DevUI.UnitTests.csproj @@ -0,0 +1,18 @@ + + + + $(TargetFrameworksCore) + false + $(NoWarn);CA1812 + + + + + + + + + + + + diff --git a/dotnet/tests/Microsoft.Agents.AI.DevUI.UnitTests/Properties/launchSettings.json b/dotnet/tests/Microsoft.Agents.AI.DevUI.UnitTests/Properties/launchSettings.json new file mode 100644 index 0000000000..783215ce29 --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.DevUI.UnitTests/Properties/launchSettings.json @@ -0,0 +1,12 @@ +{ + "profiles": { + "Microsoft.Agents.AI.DevUI.UnitTests": { + "commandName": "Project", + "launchBrowser": true, + "environmentVariables": { + "ASPNETCORE_ENVIRONMENT": "Development" + }, + "applicationUrl": "https://localhost:63009;http://localhost:63010" + } + } +} \ No newline at end of file diff --git a/dotnet/tests/Microsoft.Agents.AI.DurableTask.IntegrationTests/AgentEntityTests.cs b/dotnet/tests/Microsoft.Agents.AI.DurableTask.IntegrationTests/AgentEntityTests.cs new file mode 100644 index 0000000000..98e40ad4fb --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.DurableTask.IntegrationTests/AgentEntityTests.cs @@ -0,0 +1,140 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Diagnostics; +using System.Reflection; +using Microsoft.Agents.AI.DurableTask.State; +using Microsoft.DurableTask; +using Microsoft.DurableTask.Client; +using Microsoft.DurableTask.Client.Entities; +using Microsoft.DurableTask.Entities; +using Microsoft.Extensions.Configuration; +using OpenAI.Chat; +using Xunit.Abstractions; + +namespace Microsoft.Agents.AI.DurableTask.IntegrationTests; + +/// +/// Tests for scenarios where an external client interacts with Durable Task Agents. +/// +[Collection("Sequential")] +[Trait("Category", "Integration")] +public sealed class AgentEntityTests(ITestOutputHelper outputHelper) : IDisposable +{ + private static readonly TimeSpan s_defaultTimeout = Debugger.IsAttached + ? TimeSpan.FromMinutes(5) + : TimeSpan.FromSeconds(30); + + private static readonly IConfiguration s_configuration = + new ConfigurationBuilder() + .AddUserSecrets(Assembly.GetExecutingAssembly()) + .AddEnvironmentVariables() + .Build(); + + private readonly ITestOutputHelper _outputHelper = outputHelper; + private readonly CancellationTokenSource _cts = new(delay: s_defaultTimeout); + + private CancellationToken TestTimeoutToken => this._cts.Token; + + public void Dispose() => this._cts.Dispose(); + + [Fact] + public async Task EntityNamePrefixAsync() + { + // Setup + AIAgent simpleAgent = TestHelper.GetAzureOpenAIChatClient(s_configuration).CreateAIAgent( + name: "TestAgent", + instructions: "You are a helpful assistant that always responds with a friendly greeting." + ); + + using TestHelper testHelper = TestHelper.Start([simpleAgent], this._outputHelper); + + // A proxy agent is needed to call the hosted test agent + AIAgent simpleAgentProxy = simpleAgent.AsDurableAgentProxy(testHelper.Services); + + AgentThread thread = simpleAgentProxy.GetNewThread(); + + DurableTaskClient client = testHelper.GetClient(); + + AgentSessionId sessionId = thread.GetService(); + EntityInstanceId expectedEntityId = new($"dafx-{simpleAgent.Name}", sessionId.Key); + + EntityMetadata? entity = await client.Entities.GetEntityAsync(expectedEntityId, false, this.TestTimeoutToken); + + Assert.Null(entity); + + // Act: send a prompt to the agent + await simpleAgentProxy.RunAsync( + message: "Hello!", + thread, + cancellationToken: this.TestTimeoutToken); + + // Assert: verify the agent state was stored with the correct entity name prefix + entity = await client.Entities.GetEntityAsync(expectedEntityId, true, this.TestTimeoutToken); + + Assert.NotNull(entity); + Assert.True(entity.IncludesState); + + DurableAgentState state = entity.State.ReadAs(); + + DurableAgentStateRequest request = Assert.Single(state.Data.ConversationHistory.OfType()); + + Assert.Null(request.OrchestrationId); + } + + [Fact] + public async Task OrchestrationIdSetDuringOrchestrationAsync() + { + // Arrange + AIAgent simpleAgent = TestHelper.GetAzureOpenAIChatClient(s_configuration).CreateAIAgent( + name: "TestAgent", + instructions: "You are a helpful assistant that always responds with a friendly greeting." + ); + + using TestHelper testHelper = TestHelper.Start( + [simpleAgent], + this._outputHelper, + registry => registry.AddOrchestrator()); + + DurableTaskClient client = testHelper.GetClient(); + + // Act + string orchestrationId = await client.ScheduleNewOrchestrationInstanceAsync(nameof(TestOrchestrator), "What is the capital of Maine?"); + + OrchestrationMetadata? status = await client.WaitForInstanceCompletionAsync( + orchestrationId, + true, + this.TestTimeoutToken); + + // Assert + EntityInstanceId expectedEntityId = AgentSessionId.Parse(status.ReadOutputAs()!); + + EntityMetadata? entity = await client.Entities.GetEntityAsync(expectedEntityId, true, this.TestTimeoutToken); + + Assert.NotNull(entity); + Assert.True(entity.IncludesState); + + DurableAgentState state = entity.State.ReadAs(); + + DurableAgentStateRequest request = Assert.Single(state.Data.ConversationHistory.OfType()); + + Assert.Equal(orchestrationId, request.OrchestrationId); + } + + [System.Diagnostics.CodeAnalysis.SuppressMessage("Performance", "CA1812:Avoid uninstantiated internal classes", Justification = "Constructed via reflection.")] + private sealed class TestOrchestrator : TaskOrchestrator + { + public override async Task RunAsync(TaskOrchestrationContext context, string input) + { + DurableAIAgent writer = context.GetAgent("TestAgent"); + AgentThread writerThread = writer.GetNewThread(); + + await writer.RunAsync( + message: context.GetInput()!, + thread: writerThread); + + AgentSessionId sessionId = writerThread.GetService(); + + return sessionId.ToString(); + } + } +} diff --git a/dotnet/tests/Microsoft.Agents.AI.DurableTask.IntegrationTests/ExternalClientTests.cs b/dotnet/tests/Microsoft.Agents.AI.DurableTask.IntegrationTests/ExternalClientTests.cs new file mode 100644 index 0000000000..c43b86e330 --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.DurableTask.IntegrationTests/ExternalClientTests.cs @@ -0,0 +1,237 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.ComponentModel; +using System.Diagnostics; +using System.Reflection; +using Microsoft.Agents.AI.DurableTask.IntegrationTests.Logging; +using Microsoft.DurableTask; +using Microsoft.DurableTask.Client; +using Microsoft.Extensions.AI; +using Microsoft.Extensions.Configuration; +using OpenAI.Chat; +using Xunit.Abstractions; + +namespace Microsoft.Agents.AI.DurableTask.IntegrationTests; + +/// +/// Tests for scenarios where an external client interacts with Durable Task Agents. +/// +[Collection("Sequential")] +[Trait("Category", "Integration")] +public sealed class ExternalClientTests(ITestOutputHelper outputHelper) : IDisposable +{ + private static readonly TimeSpan s_defaultTimeout = Debugger.IsAttached + ? TimeSpan.FromMinutes(5) + : TimeSpan.FromSeconds(30); + + private static readonly IConfiguration s_configuration = + new ConfigurationBuilder() + .AddUserSecrets(Assembly.GetExecutingAssembly()) + .AddEnvironmentVariables() + .Build(); + + private readonly ITestOutputHelper _outputHelper = outputHelper; + private readonly CancellationTokenSource _cts = new(delay: s_defaultTimeout); + + private CancellationToken TestTimeoutToken => this._cts.Token; + + public void Dispose() => this._cts.Dispose(); + + [Fact] + public async Task SimplePromptAsync() + { + // Setup + AIAgent simpleAgent = TestHelper.GetAzureOpenAIChatClient(s_configuration).CreateAIAgent( + instructions: "You are a helpful assistant that always responds with a friendly greeting.", + name: "TestAgent"); + + using TestHelper testHelper = TestHelper.Start([simpleAgent], this._outputHelper); + + // A proxy agent is needed to call the hosted test agent + AIAgent simpleAgentProxy = simpleAgent.AsDurableAgentProxy(testHelper.Services); + + // Act: send a prompt to the agent and wait for a response + AgentThread thread = simpleAgentProxy.GetNewThread(); + await simpleAgentProxy.RunAsync( + message: "Hello!", + thread, + cancellationToken: this.TestTimeoutToken); + + AgentRunResponse response = await simpleAgentProxy.RunAsync( + message: "Repeat what you just said but say it like a pirate", + thread, + cancellationToken: this.TestTimeoutToken); + + // Assert: verify the agent responded appropriately + // We can't predict the exact response, but we can check that there is one response + Assert.NotNull(response); + Assert.NotEmpty(response.Text); + + // Assert: verify the expected log entries were created in the expected category + IReadOnlyCollection logs = testHelper.GetLogs(); + Assert.NotEmpty(logs); + List agentLogs = [.. logs.Where(log => log.Category.Contains(simpleAgent.Name!)).ToList()]; + Assert.NotEmpty(agentLogs); + Assert.Contains(agentLogs, log => log.EventId.Name == "LogAgentRequest" && log.Message.Contains("Hello!")); + Assert.Contains(agentLogs, log => log.EventId.Name == "LogAgentResponse"); + } + + [Fact] + public async Task CallFunctionToolsAsync() + { + int weatherToolInvocationCount = 0; + int packingListToolInvocationCount = 0; + + string GetWeather(string location) + { + weatherToolInvocationCount++; + return $"The weather in {location} is sunny with a high of 75°F and a low of 55°F."; + } + + string SuggestPackingList(string weather, bool isSunny) + { + packingListToolInvocationCount++; + return isSunny ? "Pack sunglasses and sunscreen." : "Pack a raincoat and umbrella."; + } + + AIAgent tripPlanningAgent = TestHelper.GetAzureOpenAIChatClient(s_configuration).CreateAIAgent( + instructions: "You are a trip planning assistant. Use the weather tool and packing list tool as needed.", + name: "TripPlanningAgent", + description: "An agent to help plan your day trips", + tools: [AIFunctionFactory.Create(GetWeather), AIFunctionFactory.Create(SuggestPackingList)] + ); + + using TestHelper testHelper = TestHelper.Start([tripPlanningAgent], this._outputHelper); + AIAgent tripPlanningAgentProxy = tripPlanningAgent.AsDurableAgentProxy(testHelper.Services); + + // Act: send a prompt to the agent + AgentRunResponse response = await tripPlanningAgentProxy.RunAsync( + message: "Help me figure out what to pack for my Seattle trip next Sunday", + cancellationToken: this.TestTimeoutToken); + + // Assert: verify the agent responded appropriately + // We can't predict the exact response, but we can check that there is one response + Assert.NotNull(response); + Assert.NotEmpty(response.Text); + + // Assert: verify the expected log entries were created in the expected category + IReadOnlyCollection logs = testHelper.GetLogs(); + Assert.NotEmpty(logs); + + List agentLogs = [.. logs.Where(log => log.Category.Contains(tripPlanningAgent.Name!)).ToList()]; + Assert.NotEmpty(agentLogs); + Assert.Contains(agentLogs, log => log.EventId.Name == "LogAgentRequest" && log.Message.Contains("Seattle trip")); + Assert.Contains(agentLogs, log => log.EventId.Name == "LogAgentResponse"); + + // Assert: verify the tools were called + Assert.Equal(1, weatherToolInvocationCount); + Assert.Equal(1, packingListToolInvocationCount); + } + + [Fact] + public async Task CallLongRunningFunctionToolsAsync() + { + [Description("Starts a greeting workflow and returns the workflow instance ID")] + string StartWorkflowTool(string name) + { + return DurableAgentContext.Current.ScheduleNewOrchestration(nameof(RunWorkflowAsync), input: name); + } + + [Description("Gets the current status of a previously started workflow. A null response means the workflow has not started yet.")] + static async Task GetWorkflowStatusToolAsync(string instanceId) + { + OrchestrationMetadata? status = await DurableAgentContext.Current.GetOrchestrationStatusAsync( + instanceId, + includeDetails: true); + if (status == null) + { + // If the status is not found, wait a bit before returning null to give the workflow time to start + await Task.Delay(TimeSpan.FromSeconds(1)); + } + + return status; + } + + async Task RunWorkflowAsync(TaskOrchestrationContext context, string name) + { + // 1. Get agent and create a session + DurableAIAgent agent = context.GetAgent("SimpleAgent"); + AgentThread thread = agent.GetNewThread(); + + // 2. Call an agent and tell it my name + await agent.RunAsync($"My name is {name}.", thread); + + // 3. Call the agent again with the same thread (ask it to tell me my name) + AgentRunResponse response = await agent.RunAsync("What is my name?", thread); + + return response.Text; + } + + using TestHelper testHelper = TestHelper.Start( + this._outputHelper, + configureAgents: agents => + { + // This is the agent that will be used to start the workflow + agents.AddAIAgentFactory( + "WorkflowAgent", + sp => TestHelper.GetAzureOpenAIChatClient(s_configuration).CreateAIAgent( + name: "WorkflowAgent", + instructions: "You can start greeting workflows and check their status.", + services: sp, + tools: [ + AIFunctionFactory.Create(StartWorkflowTool), + AIFunctionFactory.Create(GetWorkflowStatusToolAsync) + ])); + + // This is the agent that will be called by the workflow + agents.AddAIAgent(TestHelper.GetAzureOpenAIChatClient(s_configuration).CreateAIAgent( + name: "SimpleAgent", + instructions: "You are a simple assistant." + )); + }, + durableTaskRegistry: registry => registry.AddOrchestratorFunc(nameof(RunWorkflowAsync), RunWorkflowAsync)); + + AIAgent workflowManagerAgentProxy = testHelper.Services.GetDurableAgentProxy("WorkflowAgent"); + + // Act: send a prompt to the agent + AgentThread thread = workflowManagerAgentProxy.GetNewThread(); + await workflowManagerAgentProxy.RunAsync( + message: "Start a greeting workflow for \"John Doe\".", + thread, + cancellationToken: this.TestTimeoutToken); + + // Act: prompt it again to wait for the workflow to complete + AgentRunResponse response = await workflowManagerAgentProxy.RunAsync( + message: "Wait for the workflow to complete and tell me the result.", + thread, + cancellationToken: this.TestTimeoutToken); + + // Assert: verify the agent responded appropriately + // We can't predict the exact response, but we can check that there is one response + Assert.NotNull(response); + Assert.NotEmpty(response.Text); + Assert.Contains("John Doe", response.Text); + } + + [Fact] + public void AsDurableAgentProxy_ThrowsWhenAgentNotRegistered() + { + // Setup: Register one agent but try to use a different one + AIAgent registeredAgent = TestHelper.GetAzureOpenAIChatClient(s_configuration).CreateAIAgent( + instructions: "You are a helpful assistant.", + name: "RegisteredAgent"); + + using TestHelper testHelper = TestHelper.Start([registeredAgent], this._outputHelper); + + // Create an agent with a different name that isn't registered + AIAgent unregisteredAgent = TestHelper.GetAzureOpenAIChatClient(s_configuration).CreateAIAgent( + instructions: "You are a helpful assistant.", + name: "UnregisteredAgent"); + + // Act & Assert: Should throw AgentNotRegisteredException + AgentNotRegisteredException exception = Assert.Throws( + () => unregisteredAgent.AsDurableAgentProxy(testHelper.Services)); + + Assert.Equal("UnregisteredAgent", exception.AgentName); + } +} diff --git a/dotnet/tests/Microsoft.Agents.AI.DurableTask.IntegrationTests/Logging/LogEntry.cs b/dotnet/tests/Microsoft.Agents.AI.DurableTask.IntegrationTests/Logging/LogEntry.cs new file mode 100644 index 0000000000..fa9eddaeb4 --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.DurableTask.IntegrationTests/Logging/LogEntry.cs @@ -0,0 +1,48 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Microsoft.Extensions.Logging; + +namespace Microsoft.Agents.AI.DurableTask.IntegrationTests.Logging; + +internal sealed class LogEntry( + string category, + LogLevel level, + EventId eventId, + Exception? exception, + string message, + object? state, + IReadOnlyList> contextProperties) +{ + public string Category { get; } = category; + + public DateTime Timestamp { get; } = DateTime.Now; + + public EventId EventId { get; } = eventId; + + public LogLevel LogLevel { get; } = level; + + public Exception? Exception { get; } = exception; + + public string Message { get; } = message; + + public object? State { get; } = state; + + public IReadOnlyList> ContextProperties { get; } = contextProperties; + + public override string ToString() + { + string properties = this.ContextProperties.Count > 0 + ? $"[{string.Join(", ", this.ContextProperties.Select(kvp => $"{kvp.Key}={kvp.Value}"))}] " + : string.Empty; + + string eventName = this.EventId.Name ?? string.Empty; + string output = $"{this.Timestamp:o} [{this.Category}] {eventName} {properties}{this.Message}"; + + if (this.Exception is not null) + { + output += Environment.NewLine + this.Exception; + } + + return output; + } +} diff --git a/dotnet/tests/Microsoft.Agents.AI.DurableTask.IntegrationTests/Logging/TestLogger.cs b/dotnet/tests/Microsoft.Agents.AI.DurableTask.IntegrationTests/Logging/TestLogger.cs new file mode 100644 index 0000000000..ca80b8cf7b --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.DurableTask.IntegrationTests/Logging/TestLogger.cs @@ -0,0 +1,50 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Collections.Concurrent; +using Microsoft.Extensions.Logging; +using Xunit.Abstractions; + +namespace Microsoft.Agents.AI.DurableTask.IntegrationTests.Logging; + +internal sealed class TestLogger(string category, ITestOutputHelper output) : ILogger +{ + private readonly string _category = category; + private readonly ITestOutputHelper _output = output; + private readonly ConcurrentQueue _entries = new(); + + public IReadOnlyCollection GetLogs() => this._entries; + + public void ClearLogs() => this._entries.Clear(); + + IDisposable? ILogger.BeginScope(TState state) => null; + + bool ILogger.IsEnabled(LogLevel logLevel) => true; + + void ILogger.Log( + LogLevel logLevel, + EventId eventId, + TState state, + Exception? exception, + Func formatter) + { + LogEntry entry = new( + category: this._category, + level: logLevel, + eventId: eventId, + exception: exception, + message: formatter(state, exception), + state: state, + contextProperties: []); + + this._entries.Enqueue(entry); + + try + { + this._output.WriteLine(entry.ToString()); + } + catch (InvalidOperationException) + { + // Expected when tests are shutting down + } + } +} diff --git a/dotnet/tests/Microsoft.Agents.AI.DurableTask.IntegrationTests/Logging/TestLoggerProvider.cs b/dotnet/tests/Microsoft.Agents.AI.DurableTask.IntegrationTests/Logging/TestLoggerProvider.cs new file mode 100644 index 0000000000..7019852e5e --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.DurableTask.IntegrationTests/Logging/TestLoggerProvider.cs @@ -0,0 +1,52 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Collections.Concurrent; +using Microsoft.Extensions.Logging; +using Xunit.Abstractions; + +namespace Microsoft.Agents.AI.DurableTask.IntegrationTests.Logging; + +internal sealed class TestLoggerProvider(ITestOutputHelper output) : ILoggerProvider +{ + private readonly ITestOutputHelper _output = output ?? throw new ArgumentNullException(nameof(output)); + private readonly ConcurrentDictionary _loggers = new(StringComparer.OrdinalIgnoreCase); + + public bool TryGetLogs(string category, out IReadOnlyCollection logs) + { + if (this._loggers.TryGetValue(category, out TestLogger? logger)) + { + logs = logger.GetLogs(); + return true; + } + + logs = []; + return false; + } + + public IReadOnlyCollection GetAllLogs() + { + return this._loggers.Values + .OfType() + .SelectMany(logger => logger.GetLogs()) + .ToList() + .AsReadOnly(); + } + + public void Clear() + { + foreach (TestLogger logger in this._loggers.Values.OfType()) + { + logger.ClearLogs(); + } + } + + ILogger ILoggerProvider.CreateLogger(string categoryName) + { + return this._loggers.GetOrAdd(categoryName, _ => new TestLogger(categoryName, this._output)); + } + + void IDisposable.Dispose() + { + // no-op + } +} diff --git a/dotnet/tests/Microsoft.Agents.AI.DurableTask.IntegrationTests/Microsoft.Agents.AI.DurableTask.IntegrationTests.csproj b/dotnet/tests/Microsoft.Agents.AI.DurableTask.IntegrationTests/Microsoft.Agents.AI.DurableTask.IntegrationTests.csproj new file mode 100644 index 0000000000..db6aa6d62b --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.DurableTask.IntegrationTests/Microsoft.Agents.AI.DurableTask.IntegrationTests.csproj @@ -0,0 +1,22 @@ + + + + $(TargetFrameworksCore) + enable + b7762d10-e29b-4bb1-8b74-b6d69a667dd4 + + + + + + + + + + + + + + + + diff --git a/dotnet/tests/Microsoft.Agents.AI.DurableTask.IntegrationTests/OrchestrationTests.cs b/dotnet/tests/Microsoft.Agents.AI.DurableTask.IntegrationTests/OrchestrationTests.cs new file mode 100644 index 0000000000..0c702e6062 --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.DurableTask.IntegrationTests/OrchestrationTests.cs @@ -0,0 +1,95 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Diagnostics; +using System.Reflection; +using Microsoft.DurableTask; +using Microsoft.DurableTask.Client; +using Microsoft.Extensions.AI; +using Microsoft.Extensions.Configuration; +using OpenAI.Chat; +using Xunit.Abstractions; + +namespace Microsoft.Agents.AI.DurableTask.IntegrationTests; + +/// +/// Tests for orchestration execution scenarios with Durable Task Agents. +/// +[Collection("Sequential")] +[Trait("Category", "Integration")] +public sealed class OrchestrationTests(ITestOutputHelper outputHelper) : IDisposable +{ + private static readonly TimeSpan s_defaultTimeout = Debugger.IsAttached + ? TimeSpan.FromMinutes(5) + : TimeSpan.FromSeconds(30); + + private static readonly IConfiguration s_configuration = + new ConfigurationBuilder() + .AddUserSecrets(Assembly.GetExecutingAssembly()) + .AddEnvironmentVariables() + .Build(); + + private readonly ITestOutputHelper _outputHelper = outputHelper; + private readonly CancellationTokenSource _cts = new(delay: s_defaultTimeout); + + private CancellationToken TestTimeoutToken => this._cts.Token; + + public void Dispose() => this._cts.Dispose(); + + [Fact] + public async Task GetAgent_ThrowsWhenAgentNotRegisteredAsync() + { + // Define an orchestration that tries to use an unregistered agent + static async Task TestOrchestrationAsync(TaskOrchestrationContext context) + { + // Get an agent that hasn't been registered + DurableAIAgent agent = context.GetAgent("NonExistentAgent"); + + // This should throw when RunAsync is called because the agent doesn't exist + await agent.RunAsync("Hello"); + return "Should not reach here"; + } + + // Setup: Create test helper without registering "NonExistentAgent" + using TestHelper testHelper = TestHelper.Start( + this._outputHelper, + configureAgents: agents => + { + // Register a different agent, but not "NonExistentAgent" + agents.AddAIAgentFactory( + "OtherAgent", + sp => TestHelper.GetAzureOpenAIChatClient(s_configuration).CreateAIAgent( + name: "OtherAgent", + instructions: "You are a test agent.")); + }, + durableTaskRegistry: registry => + registry.AddOrchestratorFunc( + name: nameof(TestOrchestrationAsync), + orchestrator: TestOrchestrationAsync)); + + DurableTaskClient client = testHelper.GetClient(); + + // Act: Start the orchestration + string instanceId = await client.ScheduleNewOrchestrationInstanceAsync( + orchestratorName: nameof(TestOrchestrationAsync), + cancellation: this.TestTimeoutToken); + + // Wait for the orchestration to complete and check for failure + OrchestrationMetadata status = await client.WaitForInstanceCompletionAsync( + instanceId, + getInputsAndOutputs: true, + this.TestTimeoutToken); + + // Assert: Verify the orchestration failed with the expected exception + Assert.NotNull(status); + Assert.Equal(OrchestrationRuntimeStatus.Failed, status.RuntimeStatus); + Assert.NotNull(status.FailureDetails); + + // Verify the exception type is AgentNotRegisteredException + Assert.True( + status.FailureDetails.ErrorType == typeof(AgentNotRegisteredException).FullName, + $"Expected AgentNotRegisteredException but got ErrorType: {status.FailureDetails.ErrorType}, Message: {status.FailureDetails.ErrorMessage}"); + + // Verify the exception message contains the agent name + Assert.Contains("NonExistentAgent", status.FailureDetails.ErrorMessage, StringComparison.OrdinalIgnoreCase); + } +} diff --git a/dotnet/tests/Microsoft.Agents.AI.DurableTask.IntegrationTests/TestHelper.cs b/dotnet/tests/Microsoft.Agents.AI.DurableTask.IntegrationTests/TestHelper.cs new file mode 100644 index 0000000000..15526621d1 --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.DurableTask.IntegrationTests/TestHelper.cs @@ -0,0 +1,139 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Azure; +using Azure.AI.OpenAI; +using Azure.Identity; +using Microsoft.Agents.AI.DurableTask.IntegrationTests.Logging; +using Microsoft.DurableTask; +using Microsoft.DurableTask.Client; +using Microsoft.DurableTask.Client.AzureManaged; +using Microsoft.DurableTask.Worker; +using Microsoft.DurableTask.Worker.AzureManaged; +using Microsoft.Extensions.Configuration; +using Microsoft.Extensions.DependencyInjection; +using Microsoft.Extensions.Hosting; +using Microsoft.Extensions.Logging; +using OpenAI.Chat; +using Xunit.Abstractions; + +namespace Microsoft.Agents.AI.DurableTask.IntegrationTests; + +internal sealed class TestHelper : IDisposable +{ + private readonly TestLoggerProvider _loggerProvider; + private readonly IHost _host; + private readonly DurableTaskClient _client; + + // The static Start method should be used to create instances of this class. + private TestHelper( + TestLoggerProvider loggerProvider, + IHost host, + DurableTaskClient client) + { + this._loggerProvider = loggerProvider; + this._host = host; + this._client = client; + } + + public IServiceProvider Services => this._host.Services; + + public void Dispose() + { + this._host.Dispose(); + } + + public bool TryGetLogs(string category, out IReadOnlyCollection logs) + => this._loggerProvider.TryGetLogs(category, out logs); + + public static TestHelper Start( + AIAgent[] agents, + ITestOutputHelper outputHelper, + Action? durableTaskRegistry = null) + { + return BuildAndStartTestHelper( + outputHelper, + options => options.AddAIAgents(agents), + durableTaskRegistry); + } + + public static TestHelper Start( + ITestOutputHelper outputHelper, + Action configureAgents, + Action? durableTaskRegistry = null) + { + return BuildAndStartTestHelper( + outputHelper, + configureAgents, + durableTaskRegistry); + } + + public DurableTaskClient GetClient() => this._client; + + private static TestHelper BuildAndStartTestHelper( + ITestOutputHelper outputHelper, + Action configureAgents, + Action? durableTaskRegistry) + { + TestLoggerProvider loggerProvider = new(outputHelper); + + IHost host = Host.CreateDefaultBuilder() + .ConfigureServices((ctx, services) => + { + string dtsConnectionString = GetDurableTaskSchedulerConnectionString(ctx.Configuration); + + // Register durable agents using the caller-supplied registration action and + // apply the default chat client for agents that don't supply one themselves. + services.ConfigureDurableAgents( + options => configureAgents(options), + workerBuilder: builder => + { + builder.UseDurableTaskScheduler(dtsConnectionString); + if (durableTaskRegistry != null) + { + builder.AddTasks(durableTaskRegistry); + } + }, + clientBuilder: builder => builder.UseDurableTaskScheduler(dtsConnectionString)); + }) + .ConfigureLogging((_, logging) => + { + logging.AddProvider(loggerProvider); + logging.SetMinimumLevel(LogLevel.Debug); + }) + .Build(); + host.Start(); + + DurableTaskClient client = host.Services.GetRequiredService(); + return new TestHelper(loggerProvider, host, client); + } + + private static string GetDurableTaskSchedulerConnectionString(IConfiguration configuration) + { + // The default value is for local development using the Durable Task Scheduler emulator. + return configuration["DURABLE_TASK_SCHEDULER_CONNECTION_STRING"] + ?? "Endpoint=http://localhost:8080;TaskHub=default;Authentication=None"; + } + + internal static ChatClient GetAzureOpenAIChatClient(IConfiguration configuration) + { + string azureOpenAiEndpoint = configuration["AZURE_OPENAI_ENDPOINT"] ?? + throw new InvalidOperationException("The required AZURE_OPENAI_ENDPOINT env variable is not set."); + string azureOpenAiDeploymentName = configuration["AZURE_OPENAI_CHAT_DEPLOYMENT_NAME"] ?? + throw new InvalidOperationException("The required AZURE_OPENAI_CHAT_DEPLOYMENT_NAME env variable is not set."); + + // Check if AZURE_OPENAI_KEY is provided for key-based authentication. + // NOTE: This is not used for automated tests, but can be useful for local development. + string? azureOpenAiKey = configuration["AZURE_OPENAI_KEY"]; + + AzureOpenAIClient client = !string.IsNullOrEmpty(azureOpenAiKey) + ? new AzureOpenAIClient(new Uri(azureOpenAiEndpoint), new AzureKeyCredential(azureOpenAiKey)) + : new AzureOpenAIClient(new Uri(azureOpenAiEndpoint), new AzureCliCredential()); + + return client.GetChatClient(azureOpenAiDeploymentName); + } + + internal IReadOnlyCollection GetLogs() + { + return this._loggerProvider.GetAllLogs(); + } +} diff --git a/dotnet/tests/Microsoft.Agents.AI.DurableTask.UnitTests/AgentSessionIdTests.cs b/dotnet/tests/Microsoft.Agents.AI.DurableTask.UnitTests/AgentSessionIdTests.cs new file mode 100644 index 0000000000..03d171b7b3 --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.DurableTask.UnitTests/AgentSessionIdTests.cs @@ -0,0 +1,56 @@ +// Copyright (c) Microsoft. All rights reserved. + +using Microsoft.DurableTask.Entities; + +namespace Microsoft.Agents.AI.DurableTask.UnitTests; + +public sealed class AgentSessionIdTests +{ + [Fact] + public void ParseValidSessionId() + { + const string Name = "test-agent"; + const string Key = "12345"; + string sessionIdString = $"@dafx-{Name}@{Key}"; + AgentSessionId sessionId = AgentSessionId.Parse(sessionIdString); + + Assert.Equal(Name, sessionId.Name); + Assert.Equal(Key, sessionId.Key); + } + + [Fact] + public void ParseInvalidSessionId() + { + const string InvalidSessionIdString = "@test-agent@12345"; // Missing "dafx-" prefix + Assert.Throws(() => AgentSessionId.Parse(InvalidSessionIdString)); + } + + [Fact] + public void FromEntityId() + { + const string Name = "test-agent"; + const string Key = "12345"; + + EntityInstanceId entityId = new($"dafx-{Name}", Key); + AgentSessionId sessionId = (AgentSessionId)entityId; + + Assert.Equal(Name, sessionId.Name); + Assert.Equal(Key, sessionId.Key); + } + + [Fact] + public void FromInvalidEntityId() + { + const string Name = "test-agent"; + const string Key = "12345"; + + EntityInstanceId entityId = new(Name, Key); // Missing "dafx-" prefix + + Assert.Throws(() => + { + // This assignment should throw an exception because + // the entity ID is not a valid agent session ID. + AgentSessionId sessionId = entityId; + }); + } +} diff --git a/dotnet/tests/Microsoft.Agents.AI.DurableTask.UnitTests/DurableAgentThreadTests.cs b/dotnet/tests/Microsoft.Agents.AI.DurableTask.UnitTests/DurableAgentThreadTests.cs new file mode 100644 index 0000000000..7e5a776beb --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.DurableTask.UnitTests/DurableAgentThreadTests.cs @@ -0,0 +1,43 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json; + +namespace Microsoft.Agents.AI.DurableTask.UnitTests; + +public sealed class DurableAgentThreadTests +{ + [Fact] + public void BuiltInSerialization() + { + AgentSessionId sessionId = AgentSessionId.WithRandomKey("test-agent"); + AgentThread thread = new DurableAgentThread(sessionId); + + JsonElement serializedThread = thread.Serialize(); + + // Expected format: "{\"sessionId\":\"@dafx-test-agent@\"}" + string expectedSerializedThread = $"{{\"sessionId\":\"@dafx-{sessionId.Name}@{sessionId.Key}\"}}"; + Assert.Equal(expectedSerializedThread, serializedThread.ToString()); + + DurableAgentThread deserializedThread = DurableAgentThread.Deserialize(serializedThread); + Assert.Equal(sessionId, deserializedThread.SessionId); + } + + [Fact] + public void STJSerialization() + { + AgentSessionId sessionId = AgentSessionId.WithRandomKey("test-agent"); + AgentThread thread = new DurableAgentThread(sessionId); + + // Need to specify the type explicitly because STJ, unlike other serializers, + // does serialization based on the static type of the object, not the runtime type. + string serializedThread = JsonSerializer.Serialize(thread, typeof(DurableAgentThread)); + + // Expected format: "{\"sessionId\":\"@dafx-test-agent@\"}" + string expectedSerializedThread = $"{{\"sessionId\":\"@dafx-{sessionId.Name}@{sessionId.Key}\"}}"; + Assert.Equal(expectedSerializedThread, serializedThread); + + DurableAgentThread? deserializedThread = JsonSerializer.Deserialize(serializedThread); + Assert.NotNull(deserializedThread); + Assert.Equal(sessionId, deserializedThread.SessionId); + } +} diff --git a/dotnet/tests/Microsoft.Agents.AI.DurableTask.UnitTests/Microsoft.Agents.AI.DurableTask.UnitTests.csproj b/dotnet/tests/Microsoft.Agents.AI.DurableTask.UnitTests/Microsoft.Agents.AI.DurableTask.UnitTests.csproj new file mode 100644 index 0000000000..b0cf00cae1 --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.DurableTask.UnitTests/Microsoft.Agents.AI.DurableTask.UnitTests.csproj @@ -0,0 +1,13 @@ + + + + $(TargetFrameworksCore) + enable + b7762d10-e29b-4bb1-8b74-b6d69a667dd4 + + + + + + + diff --git a/dotnet/tests/Microsoft.Agents.AI.DurableTask.UnitTests/State/DurableAgentStateContentTests.cs b/dotnet/tests/Microsoft.Agents.AI.DurableTask.UnitTests/State/DurableAgentStateContentTests.cs new file mode 100644 index 0000000000..2fda1178e1 --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.DurableTask.UnitTests/State/DurableAgentStateContentTests.cs @@ -0,0 +1,324 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json; +using System.Text.Json.Serialization.Metadata; +using Microsoft.Agents.AI.DurableTask.State; +using Microsoft.Extensions.AI; + +namespace Microsoft.Agents.AI.DurableTask.Tests.Unit.State; + +public sealed class DurableAgentStateContentTests +{ + private static readonly JsonTypeInfo s_stateContentTypeInfo = + DurableAgentStateJsonContext.Default.GetTypeInfo(typeof(DurableAgentStateContent))!; + + [Fact] + public void ErrorContentSerializationDeserialization() + { + // Arrange + ErrorContent errorContent = new("message") + { + Details = "details", + ErrorCode = "code" + }; + + DurableAgentStateContent durableContent = DurableAgentStateContent.FromAIContent(errorContent); + + // Act + string jsonContent = JsonSerializer.Serialize(durableContent, s_stateContentTypeInfo); + + DurableAgentStateContent? convertedJsonContent = + (DurableAgentStateContent?)JsonSerializer.Deserialize(jsonContent, s_stateContentTypeInfo); + + // Assert + Assert.NotNull(convertedJsonContent); + + AIContent convertedContent = convertedJsonContent.ToAIContent(); + + ErrorContent convertedErrorContent = Assert.IsType(convertedContent); + + Assert.Equal(errorContent.Message, convertedErrorContent.Message); + Assert.Equal(errorContent.Details, convertedErrorContent.Details); + Assert.Equal(errorContent.ErrorCode, convertedErrorContent.ErrorCode); + } + + [Fact] + public void TextContentSerializationDeserialization() + { + // Arrange + TextContent textContent = new("Hello, world!"); + + DurableAgentStateContent durableContent = DurableAgentStateContent.FromAIContent(textContent); + + // Act + string jsonContent = JsonSerializer.Serialize(durableContent, s_stateContentTypeInfo); + + DurableAgentStateContent? convertedJsonContent = + (DurableAgentStateContent?)JsonSerializer.Deserialize(jsonContent, s_stateContentTypeInfo); + + // Assert + Assert.NotNull(convertedJsonContent); + + AIContent convertedContent = convertedJsonContent.ToAIContent(); + + TextContent convertedTextContent = Assert.IsType(convertedContent); + + Assert.Equal(textContent.Text, convertedTextContent.Text); + } + + [Fact] + public void FunctionCallContentSerializationDeserialization() + { + // Arrange + FunctionCallContent functionCallContent = new( + "call-123", + "MyFunction", + new Dictionary + { + { "param1", 42 }, + { "param2", "value" } + }); + + DurableAgentStateContent durableContent = DurableAgentStateContent.FromAIContent(functionCallContent); + + // Act + string jsonContent = JsonSerializer.Serialize(durableContent, s_stateContentTypeInfo); + + DurableAgentStateContent? convertedJsonContent = + (DurableAgentStateContent?)JsonSerializer.Deserialize(jsonContent, s_stateContentTypeInfo); + + // Assert + Assert.NotNull(convertedJsonContent); + + AIContent convertedContent = convertedJsonContent.ToAIContent(); + + FunctionCallContent convertedFunctionCallContent = Assert.IsType(convertedContent); + + Assert.Equal(functionCallContent.CallId, convertedFunctionCallContent.CallId); + Assert.Equal(functionCallContent.Name, convertedFunctionCallContent.Name); + + Assert.NotNull(functionCallContent.Arguments); + Assert.NotNull(convertedFunctionCallContent.Arguments); + Assert.Equal(functionCallContent.Arguments.Keys.Order(), convertedFunctionCallContent.Arguments.Keys.Order()); + + // NOTE: Deserialized dictionaries will have JSON element values rather than the original native types, + // so we only check the keys here. + foreach (string key in functionCallContent.Arguments.Keys) + { + Assert.Equal( + JsonSerializer.Serialize(functionCallContent.Arguments[key]), + JsonSerializer.Serialize(convertedFunctionCallContent.Arguments[key])); + } + } + + [Fact] + public void FunctionResultContentSerializationDeserialization() + { + // Arrange + FunctionResultContent functionResultContent = new("call-123", "return value"); + + DurableAgentStateContent durableContent = DurableAgentStateContent.FromAIContent(functionResultContent); + + // Act + string jsonContent = JsonSerializer.Serialize(durableContent, s_stateContentTypeInfo); + + DurableAgentStateContent? convertedJsonContent = + (DurableAgentStateContent?)JsonSerializer.Deserialize(jsonContent, s_stateContentTypeInfo); + + // Assert + Assert.NotNull(convertedJsonContent); + + AIContent convertedContent = convertedJsonContent.ToAIContent(); + + FunctionResultContent convertedFunctionResultContent = Assert.IsType(convertedContent); + + Assert.Equal(functionResultContent.CallId, convertedFunctionResultContent.CallId); + // NOTE: We serialize both results to JSON for comparison since deserialized objects will be + // JSON elements rather than the original native types. + Assert.Equal( + JsonSerializer.Serialize(functionResultContent.Result), + JsonSerializer.Serialize(convertedFunctionResultContent.Result)); + } + + [Theory] + [InlineData("data:text/plain;base64,SGVsbG8sIFdvcmxkIQ==", null)] // Valid data URI containing media type; pass null for separate mediaType parameter. + [InlineData("data:;base64,SGVsbG8sIFdvcmxkIQ==", "text/plain")] // Valid data URI without media type; pass media + public void DataContentSerializationDeserialization(string dataUri, string? mediaType) + { + // Arrange + DataContent dataContent = new(dataUri, mediaType); + + DurableAgentStateContent durableContent = DurableAgentStateContent.FromAIContent(dataContent); + + // Act + string jsonContent = JsonSerializer.Serialize(durableContent, s_stateContentTypeInfo); + + DurableAgentStateContent? convertedJsonContent = + (DurableAgentStateContent?)JsonSerializer.Deserialize(jsonContent, s_stateContentTypeInfo); + + // Assert + Assert.NotNull(convertedJsonContent); + + AIContent convertedContent = convertedJsonContent.ToAIContent(); + + DataContent convertedDataContent = Assert.IsType(convertedContent); + + Assert.Equal(dataContent.Uri, convertedDataContent.Uri); + Assert.Equal(dataContent.MediaType, convertedDataContent.MediaType); + } + + [Fact] + public void HostedFileContentSerializationDeserialization() + { + // Arrange + HostedFileContent hostedFileContent = new("file-123"); + + DurableAgentStateContent durableContent = DurableAgentStateContent.FromAIContent(hostedFileContent); + + // Act + string jsonContent = JsonSerializer.Serialize(durableContent, s_stateContentTypeInfo); + + DurableAgentStateContent? convertedJsonContent = + (DurableAgentStateContent?)JsonSerializer.Deserialize(jsonContent, s_stateContentTypeInfo); + + // Assert + Assert.NotNull(convertedJsonContent); + + AIContent convertedContent = convertedJsonContent.ToAIContent(); + + HostedFileContent convertedHostedFileContent = Assert.IsType(convertedContent); + + Assert.Equal(hostedFileContent.FileId, convertedHostedFileContent.FileId); + } + + [Fact] + public void HostedVectorStoreContentSerializationDeserialization() + { + // Arrange + HostedVectorStoreContent hostedVectorStoreContent = new("vs-123"); + + DurableAgentStateContent durableContent = DurableAgentStateContent.FromAIContent(hostedVectorStoreContent); + + // Act + string jsonContent = JsonSerializer.Serialize(durableContent, s_stateContentTypeInfo); + + DurableAgentStateContent? convertedJsonContent = + (DurableAgentStateContent?)JsonSerializer.Deserialize(jsonContent, s_stateContentTypeInfo); + + // Assert + Assert.NotNull(convertedJsonContent); + + AIContent convertedContent = convertedJsonContent.ToAIContent(); + + HostedVectorStoreContent convertedHostedVectorStoreContent = Assert.IsType(convertedContent); + + Assert.Equal(hostedVectorStoreContent.VectorStoreId, convertedHostedVectorStoreContent.VectorStoreId); + } + + [Fact] + public void TextReasoningContentSerializationDeserialization() + { + // Arrange + TextReasoningContent textReasoningContent = new("Reasoning chain..."); + + DurableAgentStateContent durableContent = DurableAgentStateContent.FromAIContent(textReasoningContent); + + // Act + string jsonContent = JsonSerializer.Serialize(durableContent, s_stateContentTypeInfo); + + DurableAgentStateContent? convertedJsonContent = + (DurableAgentStateContent?)JsonSerializer.Deserialize(jsonContent, s_stateContentTypeInfo); + + // Assert + Assert.NotNull(convertedJsonContent); + + AIContent convertedContent = convertedJsonContent.ToAIContent(); + + TextReasoningContent convertedTextReasoningContent = Assert.IsType(convertedContent); + + Assert.Equal(textReasoningContent.Text, convertedTextReasoningContent.Text); + } + + [Fact] + public void UriContentSerializationDeserialization() + { + // Arrange + UriContent uriContent = new(new Uri("https://example.com"), "text/html"); + + DurableAgentStateContent durableContent = DurableAgentStateContent.FromAIContent(uriContent); + + // Act + string jsonContent = JsonSerializer.Serialize(durableContent, s_stateContentTypeInfo); + + DurableAgentStateContent? convertedJsonContent = + (DurableAgentStateContent?)JsonSerializer.Deserialize(jsonContent, s_stateContentTypeInfo); + + // Assert + Assert.NotNull(convertedJsonContent); + + AIContent convertedContent = convertedJsonContent.ToAIContent(); + + UriContent convertedUriContent = Assert.IsType(convertedContent); + + Assert.Equal(uriContent.Uri, convertedUriContent.Uri); + Assert.Equal(uriContent.MediaType, convertedUriContent.MediaType); + } + + [Fact] + public void UsageContentSerializationDeserialization() + { + // Arrange + UsageDetails usageDetails = new() + { + InputTokenCount = 10, + OutputTokenCount = 5, + TotalTokenCount = 15 + }; + + UsageContent usageContent = new(usageDetails); + + DurableAgentStateContent durableContent = DurableAgentStateContent.FromAIContent(usageContent); + + // Act + string jsonContent = JsonSerializer.Serialize(durableContent, s_stateContentTypeInfo); + + DurableAgentStateContent? convertedJsonContent = + (DurableAgentStateContent?)JsonSerializer.Deserialize(jsonContent, s_stateContentTypeInfo); + + // Assert + Assert.NotNull(convertedJsonContent); + + AIContent convertedContent = convertedJsonContent.ToAIContent(); + + UsageContent convertedUsageContent = Assert.IsType(convertedContent); + + Assert.NotNull(convertedUsageContent.Details); + Assert.Equal(usageDetails.InputTokenCount, convertedUsageContent.Details.InputTokenCount); + Assert.Equal(usageDetails.OutputTokenCount, convertedUsageContent.Details.OutputTokenCount); + Assert.Equal(usageDetails.TotalTokenCount, convertedUsageContent.Details.TotalTokenCount); + } + + [Fact] + public void UnknownContentSerializationDeserialization() + { + // Arrange + TextContent originalContent = new("Some unknown content"); + + DurableAgentStateContent durableContent = DurableAgentStateUnknownContent.FromUnknownContent(originalContent); + + // Act + string jsonContent = JsonSerializer.Serialize(durableContent, s_stateContentTypeInfo); + + DurableAgentStateContent? convertedJsonContent = + (DurableAgentStateContent?)JsonSerializer.Deserialize(jsonContent, s_stateContentTypeInfo); + + // Assert + Assert.NotNull(convertedJsonContent); + + AIContent convertedContent = convertedJsonContent.ToAIContent(); + + TextContent convertedTextContent = Assert.IsType(convertedContent); + + Assert.Equal(originalContent.Text, convertedTextContent.Text); + } +} diff --git a/dotnet/tests/Microsoft.Agents.AI.DurableTask.UnitTests/State/DurableAgentStateMessageTests.cs b/dotnet/tests/Microsoft.Agents.AI.DurableTask.UnitTests/State/DurableAgentStateMessageTests.cs new file mode 100644 index 0000000000..343644d911 --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.DurableTask.UnitTests/State/DurableAgentStateMessageTests.cs @@ -0,0 +1,47 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json; +using Microsoft.Agents.AI.DurableTask.State; +using Microsoft.Extensions.AI; + +namespace Microsoft.Agents.AI.DurableTask.Tests.Unit.State; + +public sealed class DurableAgentStateMessageTests +{ + [Fact] + public void MessageSerializationDeserialization() + { + // Arrange + TextContent textContent = new("Hello, world!"); + ChatMessage message = new(ChatRole.User, [textContent]) + { + AuthorName = "User123", + CreatedAt = DateTimeOffset.UtcNow + }; + + DurableAgentStateMessage durableMessage = DurableAgentStateMessage.FromChatMessage(message); + + // Act + string jsonContent = JsonSerializer.Serialize( + durableMessage, + DurableAgentStateJsonContext.Default.GetTypeInfo(typeof(DurableAgentStateMessage))!); + + DurableAgentStateMessage? convertedJsonContent = (DurableAgentStateMessage?)JsonSerializer.Deserialize( + jsonContent, + DurableAgentStateJsonContext.Default.GetTypeInfo(typeof(DurableAgentStateMessage))!); + + // Assert + Assert.NotNull(convertedJsonContent); + + ChatMessage convertedMessage = convertedJsonContent.ToChatMessage(); + + Assert.Equal(message.AuthorName, convertedMessage.AuthorName); + Assert.Equal(message.CreatedAt, convertedMessage.CreatedAt); + Assert.Equal(message.Role, convertedMessage.Role); + + AIContent convertedContent = Assert.Single(convertedMessage.Contents); + TextContent convertedTextContent = Assert.IsType(convertedContent); + + Assert.Equal(textContent.Text, convertedTextContent.Text); + } +} diff --git a/dotnet/tests/Microsoft.Agents.AI.DurableTask.UnitTests/State/DurableAgentStateRequestTests.cs b/dotnet/tests/Microsoft.Agents.AI.DurableTask.UnitTests/State/DurableAgentStateRequestTests.cs new file mode 100644 index 0000000000..acdc602165 --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.DurableTask.UnitTests/State/DurableAgentStateRequestTests.cs @@ -0,0 +1,34 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json; +using Microsoft.Agents.AI.DurableTask.State; + +namespace Microsoft.Agents.AI.DurableTask.Tests.Unit.State; + +public sealed class DurableAgentStateRequestTests +{ + [Fact] + public void RequestSerializationDeserialization() + { + // Arrange + RunRequest originalRequest = new("Hello, world!") + { + OrchestrationId = "orch-456" + }; + DurableAgentStateRequest originalDurableRequest = DurableAgentStateRequest.FromRunRequest(originalRequest); + + // Act + string jsonContent = JsonSerializer.Serialize( + originalDurableRequest, + DurableAgentStateJsonContext.Default.GetTypeInfo(typeof(DurableAgentStateRequest))!); + + DurableAgentStateRequest? convertedJsonContent = (DurableAgentStateRequest?)JsonSerializer.Deserialize( + jsonContent, + DurableAgentStateJsonContext.Default.GetTypeInfo(typeof(DurableAgentStateRequest))!); + + // Assert + Assert.NotNull(convertedJsonContent); + Assert.Equal(originalRequest.CorrelationId, convertedJsonContent.CorrelationId); + Assert.Equal(originalRequest.OrchestrationId, convertedJsonContent.OrchestrationId); + } +} diff --git a/dotnet/tests/Microsoft.Agents.AI.DurableTask.UnitTests/State/DurableAgentStateTests.cs b/dotnet/tests/Microsoft.Agents.AI.DurableTask.UnitTests/State/DurableAgentStateTests.cs new file mode 100644 index 0000000000..f8ce5c6dec --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.DurableTask.UnitTests/State/DurableAgentStateTests.cs @@ -0,0 +1,170 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json; +using Microsoft.Agents.AI.DurableTask.State; + +namespace Microsoft.Agents.AI.DurableTask.Tests.Unit.State; + +public sealed class DurableAgentStateTests +{ + [Fact] + public void InvalidVersion() + { + // Arrange + const string JsonText = """ + { + "schemaVersion": "hello" + } + """; + + // Act & Assert + Assert.Throws( + () => JsonSerializer.Deserialize(JsonText, DurableAgentStateJsonContext.Default.DurableAgentState)); + } + + [Fact] + public void BreakingVersion() + { + // Arrange + const string JsonText = """ + { + "schemaVersion": "2.0.0" + } + """; + + // Act & Assert + Assert.Throws( + () => JsonSerializer.Deserialize(JsonText, DurableAgentStateJsonContext.Default.DurableAgentState)); + } + + [Fact] + public void MissingData() + { + // Arrange + const string JsonText = """ + { + "schemaVersion": "1.0.0" + } + """; + + // Act & Assert + Assert.Throws( + () => JsonSerializer.Deserialize(JsonText, DurableAgentStateJsonContext.Default.DurableAgentState)); + } + + [Fact] + public void ExtraData() + { + // Arrange + const string JsonText = """ + { + "schemaVersion": "1.0.0", + "data": { + "conversationHistory": [], + "extraField": "someValue" + } + } + """; + + // Act + DurableAgentState? state = JsonSerializer.Deserialize(JsonText, DurableAgentStateJsonContext.Default.DurableAgentState); + + // Assert + Assert.NotNull(state?.Data?.ExtensionData); + + Assert.True(state.Data.ExtensionData!.ContainsKey("extraField")); + Assert.Equal("someValue", state.Data.ExtensionData["extraField"]!.ToString()); + + // Act + string jsonState = JsonSerializer.Serialize(state, DurableAgentStateJsonContext.Default.DurableAgentState); + JsonDocument? jsonDocument = JsonSerializer.Deserialize(jsonState); + + // Assert + Assert.NotNull(jsonDocument); + Assert.True(jsonDocument.RootElement.TryGetProperty("data", out JsonElement dataElement)); + Assert.True(dataElement.TryGetProperty("extraField", out JsonElement extraFieldElement)); + Assert.Equal("someValue", extraFieldElement.ToString()); + } + + [Fact] + public void BasicState() + { + // Arrange + const string JsonText = """ + { + "schemaVersion": "1.0.0", + "data": { + "conversationHistory": [ + { + "$type": "request", + "correlationId": "12345", + "createdAt": "2024-01-01T12:00:00Z", + "messages": [ + { + "role": "user", + "contents": [ + { + "$type": "text", + "text": "Hello, agent!" + } + ] + } + ] + }, + { + "$type": "response", + "correlationId": "12345", + "createdAt": "2024-01-01T12:01:00Z", + "messages": [ + { + "role": "agent", + "contents": [ + { + "$type": "text", + "text": "Hi user!" + } + ] + } + ] + } + ] + } + } + """; + + // Act + DurableAgentState? state = JsonSerializer.Deserialize( + JsonText, + DurableAgentStateJsonContext.Default.DurableAgentState); + + // Assert + Assert.NotNull(state); + Assert.Equal("1.0.0", state.SchemaVersion); + Assert.NotNull(state.Data); + + Assert.Collection(state.Data.ConversationHistory, + entry => + { + Assert.IsType(entry); + Assert.Equal("12345", entry.CorrelationId); + Assert.Equal(DateTimeOffset.Parse("2024-01-01T12:00:00Z"), entry.CreatedAt); + Assert.Single(entry.Messages); + Assert.Equal("user", entry.Messages[0].Role); + DurableAgentStateContent content = Assert.Single(entry.Messages[0].Contents); + DurableAgentStateTextContent textContent = Assert.IsType(content); + Assert.Equal("Hello, agent!", textContent.Text); + }, + entry => + { + Assert.IsType(entry); + Assert.Equal("12345", entry.CorrelationId); + Assert.Equal(DateTimeOffset.Parse("2024-01-01T12:01:00Z"), entry.CreatedAt); + Assert.Single(entry.Messages); + Assert.Equal("agent", entry.Messages[0].Role); + Assert.Single(entry.Messages[0].Contents); + DurableAgentStateContent content = Assert.Single(entry.Messages[0].Contents); + DurableAgentStateTextContent textContent = Assert.IsType(content); + Assert.Equal("Hi user!", textContent.Text); + }); + } +} diff --git a/dotnet/tests/Microsoft.Agents.AI.Hosting.A2A.Tests/Microsoft.Agents.AI.Hosting.A2A.Tests.csproj b/dotnet/tests/Microsoft.Agents.AI.Hosting.A2A.Tests/Microsoft.Agents.AI.Hosting.A2A.Tests.csproj deleted file mode 100644 index 9e0a79d646..0000000000 --- a/dotnet/tests/Microsoft.Agents.AI.Hosting.A2A.Tests/Microsoft.Agents.AI.Hosting.A2A.Tests.csproj +++ /dev/null @@ -1,17 +0,0 @@ - - - - $(ProjectsCoreTargetFrameworks) - $(ProjectsDebugCoreTargetFrameworks) - - - - - - - - - - - - diff --git a/dotnet/tests/Microsoft.Agents.AI.Hosting.A2A.UnitTests/A2AIntegrationTests.cs b/dotnet/tests/Microsoft.Agents.AI.Hosting.A2A.UnitTests/A2AIntegrationTests.cs new file mode 100644 index 0000000000..48cb19789a --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.Hosting.A2A.UnitTests/A2AIntegrationTests.cs @@ -0,0 +1,87 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Text.Json; +using System.Threading.Tasks; +using A2A; +using Microsoft.Agents.AI.Hosting.A2A.UnitTests.Internal; +using Microsoft.AspNetCore.Builder; +using Microsoft.AspNetCore.Hosting.Server; +using Microsoft.AspNetCore.TestHost; +using Microsoft.Extensions.AI; +using Microsoft.Extensions.DependencyInjection; + +namespace Microsoft.Agents.AI.Hosting.A2A.UnitTests; + +public sealed class A2AIntegrationTests +{ + /// + /// Verifies that calling the A2A card endpoint with MapA2A returns an agent card with a URL populated. + /// + [Fact] + public async Task MapA2A_WithAgentCard_CardEndpointReturnsCardWithUrlAsync() + { + // Arrange + WebApplicationBuilder builder = WebApplication.CreateBuilder(); + builder.WebHost.UseTestServer(); + + IChatClient mockChatClient = new DummyChatClient(); + builder.Services.AddKeyedSingleton("chat-client", mockChatClient); + IHostedAgentBuilder agentBuilder = builder.AddAIAgent("test-agent", "Test instructions", chatClientServiceKey: "chat-client"); + builder.Services.AddLogging(); + + using WebApplication app = builder.Build(); + + var agentCard = new AgentCard + { + Name = "Test Agent", + Description = "A test agent for A2A communication", + Version = "1.0" + }; + + // Map A2A with the agent card + app.MapA2A(agentBuilder, "/a2a/test-agent", agentCard); + + await app.StartAsync(); + + try + { + // Get the test server client + TestServer testServer = app.Services.GetRequiredService() as TestServer + ?? throw new InvalidOperationException("TestServer not found"); + var httpClient = testServer.CreateClient(); + + // Act - Query the agent card endpoint + var requestUri = new Uri("/a2a/test-agent/v1/card", UriKind.Relative); + var response = await httpClient.GetAsync(requestUri); + + // Assert + Assert.True(response.IsSuccessStatusCode, $"Expected successful response but got {response.StatusCode}"); + + var content = await response.Content.ReadAsStringAsync(); + var jsonDoc = JsonDocument.Parse(content); + var root = jsonDoc.RootElement; + + // Verify the card has expected properties + Assert.True(root.TryGetProperty("name", out var nameProperty)); + Assert.Equal("Test Agent", nameProperty.GetString()); + + Assert.True(root.TryGetProperty("description", out var descProperty)); + Assert.Equal("A test agent for A2A communication", descProperty.GetString()); + + // Verify the card has a URL property and it's not null/empty + Assert.True(root.TryGetProperty("url", out var urlProperty)); + Assert.NotEqual(JsonValueKind.Null, urlProperty.ValueKind); + + var url = urlProperty.GetString(); + Assert.NotNull(url); + Assert.NotEmpty(url); + Assert.StartsWith("http", url, StringComparison.OrdinalIgnoreCase); + Assert.Equal($"{testServer.BaseAddress.ToString().TrimEnd('/')}/a2a/test-agent/v1/card", url); + } + finally + { + await app.StopAsync(); + } + } +} diff --git a/dotnet/tests/Microsoft.Agents.AI.Hosting.A2A.Tests/Converters/MessageConverterTests.cs b/dotnet/tests/Microsoft.Agents.AI.Hosting.A2A.UnitTests/Converters/MessageConverterTests.cs similarity index 97% rename from dotnet/tests/Microsoft.Agents.AI.Hosting.A2A.Tests/Converters/MessageConverterTests.cs rename to dotnet/tests/Microsoft.Agents.AI.Hosting.A2A.UnitTests/Converters/MessageConverterTests.cs index 81ce582870..69eaf3a535 100644 --- a/dotnet/tests/Microsoft.Agents.AI.Hosting.A2A.Tests/Converters/MessageConverterTests.cs +++ b/dotnet/tests/Microsoft.Agents.AI.Hosting.A2A.UnitTests/Converters/MessageConverterTests.cs @@ -5,7 +5,7 @@ using Microsoft.Agents.AI.Hosting.A2A.Converters; using Microsoft.Extensions.AI; -namespace Microsoft.Agents.AI.Hosting.A2A.Tests.Converters; +namespace Microsoft.Agents.AI.Hosting.A2A.UnitTests.Converters; public class MessageConverterTests { diff --git a/dotnet/tests/Microsoft.Agents.AI.Hosting.A2A.UnitTests/EndpointRouteA2ABuilderExtensionsTests.cs b/dotnet/tests/Microsoft.Agents.AI.Hosting.A2A.UnitTests/EndpointRouteA2ABuilderExtensionsTests.cs new file mode 100644 index 0000000000..a848528888 --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.Hosting.A2A.UnitTests/EndpointRouteA2ABuilderExtensionsTests.cs @@ -0,0 +1,479 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using A2A; +using Microsoft.Agents.AI.Hosting.A2A.UnitTests.Internal; +using Microsoft.AspNetCore.Builder; +using Microsoft.Extensions.AI; +using Microsoft.Extensions.DependencyInjection; + +namespace Microsoft.Agents.AI.Hosting.A2A.UnitTests; + +/// +/// Tests for MicrosoftAgentAIHostingA2AEndpointRouteBuilderExtensions.MapA2A method. +/// +public sealed class EndpointRouteA2ABuilderExtensionsTests +{ + /// + /// Verifies that MapA2A throws ArgumentNullException for null endpoints. + /// + [Fact] + public void MapA2A_WithAgentBuilder_NullEndpoints_ThrowsArgumentNullException() + { + // Arrange + AspNetCore.Routing.IEndpointRouteBuilder endpoints = null!; + WebApplicationBuilder builder = WebApplication.CreateBuilder(); + IChatClient mockChatClient = new DummyChatClient(); + builder.Services.AddKeyedSingleton("chat-client", mockChatClient); + IHostedAgentBuilder agentBuilder = builder.AddAIAgent("agent", "Instructions", chatClientServiceKey: "chat-client"); + + // Act & Assert + ArgumentNullException exception = Assert.Throws(() => + endpoints.MapA2A(agentBuilder, "/a2a")); + + Assert.Equal("endpoints", exception.ParamName); + } + + /// + /// Verifies that MapA2A throws ArgumentNullException for null agentBuilder. + /// + [Fact] + public void MapA2A_WithAgentBuilder_NullAgentBuilder_ThrowsArgumentNullException() + { + // Arrange + WebApplicationBuilder builder = WebApplication.CreateBuilder(); + IChatClient mockChatClient = new DummyChatClient(); + builder.Services.AddKeyedSingleton("chat-client", mockChatClient); + builder.AddAIAgent("agent", "Instructions", chatClientServiceKey: "chat-client"); + builder.Services.AddLogging(); + using WebApplication app = builder.Build(); + IHostedAgentBuilder agentBuilder = null!; + + // Act & Assert + ArgumentNullException exception = Assert.Throws(() => + app.MapA2A(agentBuilder, "/a2a")); + + Assert.Equal("agentBuilder", exception.ParamName); + } + + /// + /// Verifies that MapA2A with IHostedAgentBuilder correctly maps the agent with default task manager configuration. + /// + [Fact] + public void MapA2A_WithAgentBuilder_DefaultConfiguration_Succeeds() + { + // Arrange + WebApplicationBuilder builder = WebApplication.CreateBuilder(); + IChatClient mockChatClient = new DummyChatClient(); + builder.Services.AddKeyedSingleton("chat-client", mockChatClient); + IHostedAgentBuilder agentBuilder = builder.AddAIAgent("agent", "Instructions", chatClientServiceKey: "chat-client"); + builder.Services.AddLogging(); + using WebApplication app = builder.Build(); + + // Act & Assert - Should not throw + var result = app.MapA2A(agentBuilder, "/a2a"); + Assert.NotNull(result); + Assert.NotNull(app); + } + + /// + /// Verifies that MapA2A with IHostedAgentBuilder and custom task manager configuration succeeds. + /// + [Fact] + public void MapA2A_WithAgentBuilder_CustomTaskManagerConfiguration_Succeeds() + { + // Arrange + WebApplicationBuilder builder = WebApplication.CreateBuilder(); + IChatClient mockChatClient = new DummyChatClient(); + builder.Services.AddKeyedSingleton("chat-client", mockChatClient); + IHostedAgentBuilder agentBuilder = builder.AddAIAgent("agent", "Instructions", chatClientServiceKey: "chat-client"); + builder.Services.AddLogging(); + using WebApplication app = builder.Build(); + + // Act & Assert - Should not throw + var result = app.MapA2A(agentBuilder, "/a2a", taskManager => { }); + Assert.NotNull(result); + Assert.NotNull(app); + } + + /// + /// Verifies that MapA2A with IHostedAgentBuilder and agent card succeeds. + /// + [Fact] + public void MapA2A_WithAgentBuilder_WithAgentCard_Succeeds() + { + // Arrange + WebApplicationBuilder builder = WebApplication.CreateBuilder(); + IChatClient mockChatClient = new DummyChatClient(); + builder.Services.AddKeyedSingleton("chat-client", mockChatClient); + IHostedAgentBuilder agentBuilder = builder.AddAIAgent("agent", "Instructions", chatClientServiceKey: "chat-client"); + builder.Services.AddLogging(); + using WebApplication app = builder.Build(); + + var agentCard = new AgentCard + { + Name = "Test Agent", + Description = "A test agent for A2A communication" + }; + + // Act & Assert - Should not throw + var result = app.MapA2A(agentBuilder, "/a2a", agentCard); + Assert.NotNull(result); + Assert.NotNull(app); + } + + /// + /// Verifies that MapA2A with IHostedAgentBuilder, agent card, and custom task manager configuration succeeds. + /// + [Fact] + public void MapA2A_WithAgentBuilder_WithAgentCardAndCustomConfiguration_Succeeds() + { + // Arrange + WebApplicationBuilder builder = WebApplication.CreateBuilder(); + IChatClient mockChatClient = new DummyChatClient(); + builder.Services.AddKeyedSingleton("chat-client", mockChatClient); + IHostedAgentBuilder agentBuilder = builder.AddAIAgent("agent", "Instructions", chatClientServiceKey: "chat-client"); + builder.Services.AddLogging(); + using WebApplication app = builder.Build(); + + var agentCard = new AgentCard + { + Name = "Test Agent", + Description = "A test agent for A2A communication" + }; + + // Act & Assert - Should not throw + var result = app.MapA2A(agentBuilder, "/a2a", agentCard, taskManager => { }); + Assert.NotNull(result); + Assert.NotNull(app); + } + + /// + /// Verifies that MapA2A throws ArgumentNullException for null endpoints when using string agent name. + /// + [Fact] + public void MapA2A_WithAgentName_NullEndpoints_ThrowsArgumentNullException() + { + // Arrange + AspNetCore.Routing.IEndpointRouteBuilder endpoints = null!; + + // Act & Assert + ArgumentNullException exception = Assert.Throws(() => + endpoints.MapA2A("agent", "/a2a")); + + Assert.Equal("endpoints", exception.ParamName); + } + + /// + /// Verifies that MapA2A with string agent name correctly maps the agent. + /// + [Fact] + public void MapA2A_WithAgentName_DefaultConfiguration_Succeeds() + { + // Arrange + WebApplicationBuilder builder = WebApplication.CreateBuilder(); + IChatClient mockChatClient = new DummyChatClient(); + builder.Services.AddKeyedSingleton("chat-client", mockChatClient); + builder.AddAIAgent("agent", "Instructions", chatClientServiceKey: "chat-client"); + builder.Services.AddLogging(); + using WebApplication app = builder.Build(); + + // Act & Assert - Should not throw + var result = app.MapA2A("agent", "/a2a"); + Assert.NotNull(result); + Assert.NotNull(app); + } + + /// + /// Verifies that MapA2A with string agent name and custom task manager configuration succeeds. + /// + [Fact] + public void MapA2A_WithAgentName_CustomTaskManagerConfiguration_Succeeds() + { + // Arrange + WebApplicationBuilder builder = WebApplication.CreateBuilder(); + IChatClient mockChatClient = new DummyChatClient(); + builder.Services.AddKeyedSingleton("chat-client", mockChatClient); + builder.AddAIAgent("agent", "Instructions", chatClientServiceKey: "chat-client"); + builder.Services.AddLogging(); + using WebApplication app = builder.Build(); + + // Act & Assert - Should not throw + var result = app.MapA2A("agent", "/a2a", taskManager => { }); + Assert.NotNull(result); + Assert.NotNull(app); + } + + /// + /// Verifies that MapA2A with string agent name and agent card succeeds. + /// + [Fact] + public void MapA2A_WithAgentName_WithAgentCard_Succeeds() + { + // Arrange + WebApplicationBuilder builder = WebApplication.CreateBuilder(); + IChatClient mockChatClient = new DummyChatClient(); + builder.Services.AddKeyedSingleton("chat-client", mockChatClient); + builder.AddAIAgent("agent", "Instructions", chatClientServiceKey: "chat-client"); + builder.Services.AddLogging(); + using WebApplication app = builder.Build(); + + var agentCard = new AgentCard + { + Name = "Test Agent", + Description = "A test agent for A2A communication" + }; + + // Act & Assert - Should not throw + var result = app.MapA2A("agent", "/a2a", agentCard); + Assert.NotNull(result); + Assert.NotNull(app); + } + + /// + /// Verifies that MapA2A with string agent name, agent card, and custom task manager configuration succeeds. + /// + [Fact] + public void MapA2A_WithAgentName_WithAgentCardAndCustomConfiguration_Succeeds() + { + // Arrange + WebApplicationBuilder builder = WebApplication.CreateBuilder(); + IChatClient mockChatClient = new DummyChatClient(); + builder.Services.AddKeyedSingleton("chat-client", mockChatClient); + builder.AddAIAgent("agent", "Instructions", chatClientServiceKey: "chat-client"); + builder.Services.AddLogging(); + using WebApplication app = builder.Build(); + + var agentCard = new AgentCard + { + Name = "Test Agent", + Description = "A test agent for A2A communication" + }; + + // Act & Assert - Should not throw + var result = app.MapA2A("agent", "/a2a", agentCard, taskManager => { }); + Assert.NotNull(result); + Assert.NotNull(app); + } + + /// + /// Verifies that MapA2A throws ArgumentNullException for null endpoints when using AIAgent. + /// + [Fact] + public void MapA2A_WithAIAgent_NullEndpoints_ThrowsArgumentNullException() + { + // Arrange + AspNetCore.Routing.IEndpointRouteBuilder endpoints = null!; + + // Act & Assert + ArgumentNullException exception = Assert.Throws(() => + endpoints.MapA2A((AIAgent)null!, "/a2a")); + + Assert.Equal("endpoints", exception.ParamName); + } + + /// + /// Verifies that MapA2A with AIAgent correctly maps the agent. + /// + [Fact] + public void MapA2A_WithAIAgent_DefaultConfiguration_Succeeds() + { + // Arrange + WebApplicationBuilder builder = WebApplication.CreateBuilder(); + IChatClient mockChatClient = new DummyChatClient(); + builder.Services.AddKeyedSingleton("chat-client", mockChatClient); + builder.AddAIAgent("agent", "Instructions", chatClientServiceKey: "chat-client"); + builder.Services.AddLogging(); + using WebApplication app = builder.Build(); + AIAgent agent = app.Services.GetRequiredKeyedService("agent"); + + // Act & Assert - Should not throw + var result = app.MapA2A(agent, "/a2a"); + Assert.NotNull(result); + Assert.NotNull(app); + } + + /// + /// Verifies that MapA2A with AIAgent and custom task manager configuration succeeds. + /// + [Fact] + public void MapA2A_WithAIAgent_CustomTaskManagerConfiguration_Succeeds() + { + // Arrange + WebApplicationBuilder builder = WebApplication.CreateBuilder(); + IChatClient mockChatClient = new DummyChatClient(); + builder.Services.AddKeyedSingleton("chat-client", mockChatClient); + builder.AddAIAgent("agent", "Instructions", chatClientServiceKey: "chat-client"); + builder.Services.AddLogging(); + using WebApplication app = builder.Build(); + AIAgent agent = app.Services.GetRequiredKeyedService("agent"); + + // Act & Assert - Should not throw + var result = app.MapA2A(agent, "/a2a", taskManager => { }); + Assert.NotNull(result); + Assert.NotNull(app); + } + + /// + /// Verifies that MapA2A with AIAgent and agent card succeeds. + /// + [Fact] + public void MapA2A_WithAIAgent_WithAgentCard_Succeeds() + { + // Arrange + WebApplicationBuilder builder = WebApplication.CreateBuilder(); + IChatClient mockChatClient = new DummyChatClient(); + builder.Services.AddKeyedSingleton("chat-client", mockChatClient); + builder.AddAIAgent("agent", "Instructions", chatClientServiceKey: "chat-client"); + builder.Services.AddLogging(); + using WebApplication app = builder.Build(); + AIAgent agent = app.Services.GetRequiredKeyedService("agent"); + + var agentCard = new AgentCard + { + Name = "Test Agent", + Description = "A test agent for A2A communication" + }; + + // Act & Assert - Should not throw + var result = app.MapA2A(agent, "/a2a", agentCard); + Assert.NotNull(result); + Assert.NotNull(app); + } + + /// + /// Verifies that MapA2A with AIAgent, agent card, and custom task manager configuration succeeds. + /// + [Fact] + public void MapA2A_WithAIAgent_WithAgentCardAndCustomConfiguration_Succeeds() + { + // Arrange + WebApplicationBuilder builder = WebApplication.CreateBuilder(); + IChatClient mockChatClient = new DummyChatClient(); + builder.Services.AddKeyedSingleton("chat-client", mockChatClient); + builder.AddAIAgent("agent", "Instructions", chatClientServiceKey: "chat-client"); + builder.Services.AddLogging(); + using WebApplication app = builder.Build(); + AIAgent agent = app.Services.GetRequiredKeyedService("agent"); + + var agentCard = new AgentCard + { + Name = "Test Agent", + Description = "A test agent for A2A communication" + }; + + // Act & Assert - Should not throw + var result = app.MapA2A(agent, "/a2a", agentCard, taskManager => { }); + Assert.NotNull(result); + Assert.NotNull(app); + } + + /// + /// Verifies that MapA2A throws ArgumentNullException for null endpoints when using ITaskManager. + /// + [Fact] + public void MapA2A_WithTaskManager_NullEndpoints_ThrowsArgumentNullException() + { + // Arrange + AspNetCore.Routing.IEndpointRouteBuilder endpoints = null!; + ITaskManager taskManager = null!; + + // Act & Assert + ArgumentNullException exception = Assert.Throws(() => + endpoints.MapA2A(taskManager, "/a2a")); + + Assert.Equal("endpoints", exception.ParamName); + } + + /// + /// Verifies that multiple agents can be mapped to different paths. + /// + [Fact] + public void MapA2A_MultipleAgents_Succeeds() + { + // Arrange + WebApplicationBuilder builder = WebApplication.CreateBuilder(); + IChatClient mockChatClient = new DummyChatClient(); + builder.Services.AddKeyedSingleton("chat-client", mockChatClient); + IHostedAgentBuilder agent1Builder = builder.AddAIAgent("agent1", "Instructions1", chatClientServiceKey: "chat-client"); + IHostedAgentBuilder agent2Builder = builder.AddAIAgent("agent2", "Instructions2", chatClientServiceKey: "chat-client"); + builder.Services.AddLogging(); + using WebApplication app = builder.Build(); + + // Act & Assert - Should not throw + app.MapA2A(agent1Builder, "/a2a/agent1"); + app.MapA2A(agent2Builder, "/a2a/agent2"); + Assert.NotNull(app); + } + + /// + /// Verifies that custom paths can be specified for A2A endpoints. + /// + [Fact] + public void MapA2A_WithCustomPath_AcceptsValidPath() + { + // Arrange + WebApplicationBuilder builder = WebApplication.CreateBuilder(); + IChatClient mockChatClient = new DummyChatClient(); + builder.Services.AddKeyedSingleton("chat-client", mockChatClient); + IHostedAgentBuilder agentBuilder = builder.AddAIAgent("agent", "Instructions", chatClientServiceKey: "chat-client"); + builder.Services.AddLogging(); + using WebApplication app = builder.Build(); + + // Act & Assert - Should not throw + app.MapA2A(agentBuilder, "/custom/a2a/path"); + Assert.NotNull(app); + } + + /// + /// Verifies that task manager configuration callback is invoked correctly. + /// + [Fact] + public void MapA2A_WithAgentBuilder_TaskManagerConfigurationCallbackInvoked() + { + // Arrange + WebApplicationBuilder builder = WebApplication.CreateBuilder(); + IChatClient mockChatClient = new DummyChatClient(); + builder.Services.AddKeyedSingleton("chat-client", mockChatClient); + IHostedAgentBuilder agentBuilder = builder.AddAIAgent("agent", "Instructions", chatClientServiceKey: "chat-client"); + builder.Services.AddLogging(); + using WebApplication app = builder.Build(); + + bool configureCallbackInvoked = false; + + // Act + app.MapA2A(agentBuilder, "/a2a", taskManager => + { + configureCallbackInvoked = true; + Assert.NotNull(taskManager); + }); + + // Assert + Assert.True(configureCallbackInvoked); + } + + /// + /// Verifies that agent card with all properties is accepted. + /// + [Fact] + public void MapA2A_WithAgentBuilder_FullAgentCard_Succeeds() + { + // Arrange + WebApplicationBuilder builder = WebApplication.CreateBuilder(); + IChatClient mockChatClient = new DummyChatClient(); + builder.Services.AddKeyedSingleton("chat-client", mockChatClient); + IHostedAgentBuilder agentBuilder = builder.AddAIAgent("agent", "Instructions", chatClientServiceKey: "chat-client"); + builder.Services.AddLogging(); + using WebApplication app = builder.Build(); + + var agentCard = new AgentCard + { + Name = "Test Agent", + Description = "A comprehensive test agent" + }; + + // Act & Assert - Should not throw + var result = app.MapA2A(agentBuilder, "/a2a", agentCard); + Assert.NotNull(result); + } +} diff --git a/dotnet/tests/Microsoft.Agents.AI.Hosting.A2A.UnitTests/Internal/DummyChatClient.cs b/dotnet/tests/Microsoft.Agents.AI.Hosting.A2A.UnitTests/Internal/DummyChatClient.cs new file mode 100644 index 0000000000..efab140b68 --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.Hosting.A2A.UnitTests/Internal/DummyChatClient.cs @@ -0,0 +1,30 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Collections.Generic; +using System.Threading; +using System.Threading.Tasks; +using Microsoft.Extensions.AI; + +namespace Microsoft.Agents.AI.Hosting.A2A.UnitTests.Internal; + +internal sealed class DummyChatClient : IChatClient +{ + public void Dispose() + { + throw new NotImplementedException(); + } + + public Task GetResponseAsync(IEnumerable messages, ChatOptions? options = null, CancellationToken cancellationToken = default) + { + throw new NotImplementedException(); + } + + public object? GetService(Type serviceType, object? serviceKey = null) => + serviceType.IsInstanceOfType(this) ? this : null; + + public IAsyncEnumerable GetStreamingResponseAsync(IEnumerable messages, ChatOptions? options = null, CancellationToken cancellationToken = default) + { + throw new NotImplementedException(); + } +} diff --git a/dotnet/tests/Microsoft.Agents.AI.Hosting.A2A.UnitTests/Microsoft.Agents.AI.Hosting.A2A.UnitTests.csproj b/dotnet/tests/Microsoft.Agents.AI.Hosting.A2A.UnitTests/Microsoft.Agents.AI.Hosting.A2A.UnitTests.csproj new file mode 100644 index 0000000000..42d8682870 --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.Hosting.A2A.UnitTests/Microsoft.Agents.AI.Hosting.A2A.UnitTests.csproj @@ -0,0 +1,21 @@ + + + + $(TargetFrameworksCore) + + + + + + + + + + + + + + + + + diff --git a/dotnet/tests/Microsoft.Agents.AI.Hosting.A2A.UnitTests/Properties/launchSettings.json b/dotnet/tests/Microsoft.Agents.AI.Hosting.A2A.UnitTests/Properties/launchSettings.json new file mode 100644 index 0000000000..6b8f8d04a4 --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.Hosting.A2A.UnitTests/Properties/launchSettings.json @@ -0,0 +1,12 @@ +{ + "profiles": { + "Microsoft.Agents.AI.Hosting.A2A.UnitTests": { + "commandName": "Project", + "launchBrowser": true, + "environmentVariables": { + "ASPNETCORE_ENVIRONMENT": "Development" + }, + "applicationUrl": "https://localhost:52186;http://localhost:52187" + } + } +} \ No newline at end of file diff --git a/dotnet/tests/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.IntegrationTests/BasicStreamingTests.cs b/dotnet/tests/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.IntegrationTests/BasicStreamingTests.cs index fdf8c6abad..69560421cf 100644 --- a/dotnet/tests/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.IntegrationTests/BasicStreamingTests.cs +++ b/dotnet/tests/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.IntegrationTests/BasicStreamingTests.cs @@ -29,8 +29,9 @@ public async Task ClientReceivesStreamedAssistantMessageAsync() { // Arrange await this.SetupTestServerAsync(); - AGUIAgent agent = new("assistant", "Sample assistant", this._client!, ""); - AgentThread thread = agent.GetNewThread(); + var chatClient = new AGUIChatClient(this._client!, "", null); + AIAgent agent = chatClient.CreateAIAgent(instructions: null, name: "assistant", description: "Sample assistant", tools: []); + ChatClientAgentThread thread = (ChatClientAgentThread)agent.GetNewThread(); ChatMessage userMessage = new(ChatRole.User, "hello"); List updates = []; @@ -42,16 +43,16 @@ public async Task ClientReceivesStreamedAssistantMessageAsync() } // Assert - InMemoryAgentThread? inMemoryThread = thread.GetService(); - inMemoryThread.Should().NotBeNull(); - inMemoryThread!.MessageStore.Should().HaveCount(2); - inMemoryThread.MessageStore[0].Role.Should().Be(ChatRole.User); - inMemoryThread.MessageStore[0].Text.Should().Be("hello"); - inMemoryThread.MessageStore[1].Role.Should().Be(ChatRole.Assistant); - inMemoryThread.MessageStore[1].Text.Should().Be("Hello from fake agent!"); + thread.Should().NotBeNull(); updates.Should().NotBeEmpty(); updates.Should().AllSatisfy(u => u.Role.Should().Be(ChatRole.Assistant)); + + // Verify assistant response message + AgentRunResponse response = updates.ToAgentRunResponse(); + response.Messages.Should().HaveCount(1); + response.Messages[0].Role.Should().Be(ChatRole.Assistant); + response.Messages[0].Text.Should().Be("Hello from fake agent!"); } [Fact] @@ -59,8 +60,9 @@ public async Task ClientReceivesRunLifecycleEventsAsync() { // Arrange await this.SetupTestServerAsync(); - AGUIAgent agent = new("assistant", "Sample assistant", this._client!, ""); - AgentThread thread = agent.GetNewThread(); + var chatClient = new AGUIChatClient(this._client!, "", null); + AIAgent agent = chatClient.CreateAIAgent(instructions: null, name: "assistant", description: "Sample assistant", tools: []); + ChatClientAgentThread thread = (ChatClientAgentThread)agent.GetNewThread(); ChatMessage userMessage = new(ChatRole.User, "test"); List updates = []; @@ -102,8 +104,9 @@ public async Task RunAsyncAggregatesStreamingUpdatesAsync() { // Arrange await this.SetupTestServerAsync(); - AGUIAgent agent = new("assistant", "Sample assistant", this._client!, ""); - AgentThread thread = agent.GetNewThread(); + var chatClient = new AGUIChatClient(this._client!, "", null); + AIAgent agent = chatClient.CreateAIAgent(instructions: null, name: "assistant", description: "Sample assistant", tools: []); + ChatClientAgentThread thread = (ChatClientAgentThread)agent.GetNewThread(); ChatMessage userMessage = new(ChatRole.User, "hello"); // Act @@ -120,13 +123,14 @@ public async Task MultiTurnConversationPreservesAllMessagesInThreadAsync() { // Arrange await this.SetupTestServerAsync(); - AGUIAgent agent = new("assistant", "Sample assistant", this._client!, ""); - AgentThread thread = agent.GetNewThread(); + var chatClient = new AGUIChatClient(this._client!, "", null); + AIAgent agent = chatClient.CreateAIAgent(instructions: null, name: "assistant", description: "Sample assistant", tools: []); + ChatClientAgentThread chatClientThread = (ChatClientAgentThread)agent.GetNewThread(); ChatMessage firstUserMessage = new(ChatRole.User, "First question"); // Act - First turn List firstTurnUpdates = []; - await foreach (AgentRunResponseUpdate update in agent.RunStreamingAsync([firstUserMessage], thread, new AgentRunOptions(), CancellationToken.None)) + await foreach (AgentRunResponseUpdate update in agent.RunStreamingAsync([firstUserMessage], chatClientThread, new AgentRunOptions(), CancellationToken.None)) { firstTurnUpdates.Add(update); } @@ -137,7 +141,7 @@ public async Task MultiTurnConversationPreservesAllMessagesInThreadAsync() // Act - Second turn with another message ChatMessage secondUserMessage = new(ChatRole.User, "Second question"); List secondTurnUpdates = []; - await foreach (AgentRunResponseUpdate update in agent.RunStreamingAsync([secondUserMessage], thread, new AgentRunOptions(), CancellationToken.None)) + await foreach (AgentRunResponseUpdate update in agent.RunStreamingAsync([secondUserMessage], chatClientThread, new AgentRunOptions(), CancellationToken.None)) { secondTurnUpdates.Add(update); } @@ -145,23 +149,17 @@ public async Task MultiTurnConversationPreservesAllMessagesInThreadAsync() // Assert second turn completed secondTurnUpdates.Should().Contain(u => !string.IsNullOrEmpty(u.Text)); - // Assert - Thread should contain all 4 messages (2 user + 2 assistant) - InMemoryAgentThread? inMemoryThread = thread.GetService(); - inMemoryThread.Should().NotBeNull(); - inMemoryThread!.MessageStore.Should().HaveCount(4); - - // Verify message order and content - inMemoryThread.MessageStore[0].Role.Should().Be(ChatRole.User); - inMemoryThread.MessageStore[0].Text.Should().Be("First question"); - - inMemoryThread.MessageStore[1].Role.Should().Be(ChatRole.Assistant); - inMemoryThread.MessageStore[1].Text.Should().Be("Hello from fake agent!"); - - inMemoryThread.MessageStore[2].Role.Should().Be(ChatRole.User); - inMemoryThread.MessageStore[2].Text.Should().Be("Second question"); - - inMemoryThread.MessageStore[3].Role.Should().Be(ChatRole.Assistant); - inMemoryThread.MessageStore[3].Text.Should().Be("Hello from fake agent!"); + // Verify first turn assistant response + AgentRunResponse firstResponse = firstTurnUpdates.ToAgentRunResponse(); + firstResponse.Messages.Should().HaveCount(1); + firstResponse.Messages[0].Role.Should().Be(ChatRole.Assistant); + firstResponse.Messages[0].Text.Should().Be("Hello from fake agent!"); + + // Verify second turn assistant response + AgentRunResponse secondResponse = secondTurnUpdates.ToAgentRunResponse(); + secondResponse.Messages.Should().HaveCount(1); + secondResponse.Messages[0].Role.Should().Be(ChatRole.Assistant); + secondResponse.Messages[0].Text.Should().Be("Hello from fake agent!"); } [Fact] @@ -169,14 +167,15 @@ public async Task AgentSendsMultipleMessagesInOneTurnAsync() { // Arrange await this.SetupTestServerAsync(useMultiMessageAgent: true); - AGUIAgent agent = new("assistant", "Sample assistant", this._client!, ""); - AgentThread thread = agent.GetNewThread(); + var chatClient = new AGUIChatClient(this._client!, "", null); + AIAgent agent = chatClient.CreateAIAgent(instructions: null, name: "assistant", description: "Sample assistant", tools: []); + ChatClientAgentThread chatClientThread = (ChatClientAgentThread)agent.GetNewThread(); ChatMessage userMessage = new(ChatRole.User, "Tell me a story"); List updates = []; // Act - await foreach (AgentRunResponseUpdate update in agent.RunStreamingAsync([userMessage], thread, new AgentRunOptions(), CancellationToken.None)) + await foreach (AgentRunResponseUpdate update in agent.RunStreamingAsync([userMessage], chatClientThread, new AgentRunOptions(), CancellationToken.None)) { updates.Add(update); } @@ -189,12 +188,10 @@ public async Task AgentSendsMultipleMessagesInOneTurnAsync() List messageIds = textUpdates.Select(u => u.MessageId).Where(id => !string.IsNullOrEmpty(id)).Distinct().ToList()!; messageIds.Should().HaveCountGreaterThan(1, "agent should send multiple messages"); - // Verify thread contains user message plus multiple assistant messages - InMemoryAgentThread? inMemoryThread = thread.GetService(); - inMemoryThread.Should().NotBeNull(); - inMemoryThread!.MessageStore.Should().HaveCountGreaterThan(2); - inMemoryThread.MessageStore[0].Role.Should().Be(ChatRole.User); - inMemoryThread.MessageStore.Skip(1).Should().AllSatisfy(m => m.Role.Should().Be(ChatRole.Assistant)); + // Verify assistant messages from updates + AgentRunResponse response = updates.ToAgentRunResponse(); + response.Messages.Should().HaveCountGreaterThan(1); + response.Messages.Should().AllSatisfy(m => m.Role.Should().Be(ChatRole.Assistant)); } [Fact] @@ -202,8 +199,9 @@ public async Task UserSendsMultipleMessagesAtOnceAsync() { // Arrange await this.SetupTestServerAsync(); - AGUIAgent agent = new("assistant", "Sample assistant", this._client!, ""); - AgentThread thread = agent.GetNewThread(); + var chatClient = new AGUIChatClient(this._client!, "", null); + AIAgent agent = chatClient.CreateAIAgent(instructions: null, name: "assistant", description: "Sample assistant", tools: []); + ChatClientAgentThread chatClientThread = (ChatClientAgentThread)agent.GetNewThread(); // Multiple user messages sent in one turn ChatMessage[] userMessages = @@ -216,30 +214,20 @@ public async Task UserSendsMultipleMessagesAtOnceAsync() List updates = []; // Act - await foreach (AgentRunResponseUpdate update in agent.RunStreamingAsync(userMessages, thread, new AgentRunOptions(), CancellationToken.None)) + await foreach (AgentRunResponseUpdate update in agent.RunStreamingAsync(userMessages, chatClientThread, new AgentRunOptions(), CancellationToken.None)) { updates.Add(update); } // Assert - Should have received assistant response updates.Should().Contain(u => !string.IsNullOrEmpty(u.Text)); + updates.Should().Contain(u => u.Role == ChatRole.Assistant); - // Verify thread contains all user messages plus assistant response - InMemoryAgentThread? inMemoryThread = thread.GetService(); - inMemoryThread.Should().NotBeNull(); - inMemoryThread!.MessageStore.Should().HaveCount(4); // 3 user + 1 assistant - - inMemoryThread.MessageStore[0].Role.Should().Be(ChatRole.User); - inMemoryThread.MessageStore[0].Text.Should().Be("First part of question"); - - inMemoryThread.MessageStore[1].Role.Should().Be(ChatRole.User); - inMemoryThread.MessageStore[1].Text.Should().Be("Second part of question"); - - inMemoryThread.MessageStore[2].Role.Should().Be(ChatRole.User); - inMemoryThread.MessageStore[2].Text.Should().Be("Third part of question"); - - inMemoryThread.MessageStore[3].Role.Should().Be(ChatRole.Assistant); - inMemoryThread.MessageStore[3].Text.Should().Be("Hello from fake agent!"); + // Verify assistant response message + AgentRunResponse response = updates.ToAgentRunResponse(); + response.Messages.Should().HaveCount(1); + response.Messages[0].Role.Should().Be(ChatRole.Assistant); + response.Messages[0].Text.Should().Be("Hello from fake agent!"); } private async Task SetupTestServerAsync(bool useMultiMessageAgent = false) @@ -247,6 +235,8 @@ private async Task SetupTestServerAsync(bool useMultiMessageAgent = false) WebApplicationBuilder builder = WebApplication.CreateBuilder(); builder.WebHost.UseTestServer(); + builder.Services.AddAGUI(); + if (useMultiMessageAgent) { builder.Services.AddSingleton(); @@ -286,18 +276,9 @@ public async ValueTask DisposeAsync() [SuppressMessage("Performance", "CA1812:Avoid uninstantiated internal classes", Justification = "Instantiated via dependency injection")] internal sealed class FakeChatClientAgent : AIAgent { - private readonly string _agentId; - private readonly string _description; - - public FakeChatClientAgent() - { - this._agentId = "fake-agent"; - this._description = "A fake agent for testing"; - } - - public override string Id => this._agentId; + protected override string? IdCore => "fake-agent"; - public override string? Description => this._description; + public override string? Description => "A fake agent for testing"; public override AgentThread GetNewThread() { @@ -363,18 +344,9 @@ public FakeInMemoryAgentThread(JsonElement serializedThread, JsonSerializerOptio [SuppressMessage("Performance", "CA1812:Avoid uninstantiated internal classes", Justification = "Instantiated via dependency injection")] internal sealed class FakeMultiMessageAgent : AIAgent { - private readonly string _agentId; - private readonly string _description; + protected override string? IdCore => "fake-multi-message-agent"; - public FakeMultiMessageAgent() - { - this._agentId = "fake-multi-message-agent"; - this._description = "A fake agent that sends multiple messages for testing"; - } - - public override string Id => this._agentId; - - public override string? Description => this._description; + public override string? Description => "A fake agent that sends multiple messages for testing"; public override AgentThread GetNewThread() { @@ -462,4 +434,6 @@ public FakeInMemoryAgentThread(JsonElement serializedThread, JsonSerializerOptio { } } + + public override object? GetService(Type serviceType, object? serviceKey = null) => null; } diff --git a/dotnet/tests/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.IntegrationTests/ForwardedPropertiesTests.cs b/dotnet/tests/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.IntegrationTests/ForwardedPropertiesTests.cs new file mode 100644 index 0000000000..df8caea214 --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.IntegrationTests/ForwardedPropertiesTests.cs @@ -0,0 +1,358 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Collections.Generic; +using System.Diagnostics.CodeAnalysis; +using System.IO; +using System.Net.Http; +using System.Net.ServerSentEvents; +using System.Runtime.CompilerServices; +using System.Text; +using System.Text.Json; +using System.Threading; +using System.Threading.Tasks; +using FluentAssertions; +using Microsoft.AspNetCore.Builder; +using Microsoft.AspNetCore.Hosting.Server; +using Microsoft.AspNetCore.TestHost; +using Microsoft.Extensions.AI; +using Microsoft.Extensions.DependencyInjection; + +namespace Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.IntegrationTests; + +public sealed class ForwardedPropertiesTests : IAsyncDisposable +{ + private WebApplication? _app; + private HttpClient? _client; + + [Fact] + public async Task ForwardedProps_AreParsedAndPassedToAgent_WhenProvidedInRequestAsync() + { + // Arrange + FakeForwardedPropsAgent fakeAgent = new(); + await this.SetupTestServerAsync(fakeAgent); + + // Create request JSON with forwardedProps (per AG-UI protocol spec) + const string RequestJson = """ + { + "threadId": "thread-123", + "runId": "run-456", + "messages": [{ "id": "msg-1", "role": "user", "content": "test forwarded props" }], + "forwardedProps": { "customProp": "customValue", "sessionId": "test-session-123" } + } + """; + + using StringContent content = new(RequestJson, Encoding.UTF8, "application/json"); + + // Act + HttpResponseMessage response = await this._client!.PostAsync(new Uri("/agent", UriKind.Relative), content); + + // Assert + response.IsSuccessStatusCode.Should().BeTrue(); + fakeAgent.ReceivedForwardedProperties.ValueKind.Should().Be(JsonValueKind.Object); + fakeAgent.ReceivedForwardedProperties.GetProperty("customProp").GetString().Should().Be("customValue"); + fakeAgent.ReceivedForwardedProperties.GetProperty("sessionId").GetString().Should().Be("test-session-123"); + } + + [Fact] + public async Task ForwardedProps_WithNestedObjects_AreCorrectlyParsedAsync() + { + // Arrange + FakeForwardedPropsAgent fakeAgent = new(); + await this.SetupTestServerAsync(fakeAgent); + + const string RequestJson = """ + { + "threadId": "thread-123", + "runId": "run-456", + "messages": [{ "id": "msg-1", "role": "user", "content": "test nested props" }], + "forwardedProps": { + "user": { "id": "user-1", "name": "Test User" }, + "metadata": { "version": "1.0", "feature": "test" } + } + } + """; + + using StringContent content = new(RequestJson, Encoding.UTF8, "application/json"); + + // Act + HttpResponseMessage response = await this._client!.PostAsync(new Uri("/agent", UriKind.Relative), content); + + // Assert + response.IsSuccessStatusCode.Should().BeTrue(); + fakeAgent.ReceivedForwardedProperties.ValueKind.Should().Be(JsonValueKind.Object); + + JsonElement user = fakeAgent.ReceivedForwardedProperties.GetProperty("user"); + user.GetProperty("id").GetString().Should().Be("user-1"); + user.GetProperty("name").GetString().Should().Be("Test User"); + + JsonElement metadata = fakeAgent.ReceivedForwardedProperties.GetProperty("metadata"); + metadata.GetProperty("version").GetString().Should().Be("1.0"); + metadata.GetProperty("feature").GetString().Should().Be("test"); + } + + [Fact] + public async Task ForwardedProps_WithArrays_AreCorrectlyParsedAsync() + { + // Arrange + FakeForwardedPropsAgent fakeAgent = new(); + await this.SetupTestServerAsync(fakeAgent); + + const string RequestJson = """ + { + "threadId": "thread-123", + "runId": "run-456", + "messages": [{ "id": "msg-1", "role": "user", "content": "test array props" }], + "forwardedProps": { + "tags": ["tag1", "tag2", "tag3"], + "scores": [1, 2, 3, 4, 5] + } + } + """; + + using StringContent content = new(RequestJson, Encoding.UTF8, "application/json"); + + // Act + HttpResponseMessage response = await this._client!.PostAsync(new Uri("/agent", UriKind.Relative), content); + + // Assert + response.IsSuccessStatusCode.Should().BeTrue(); + fakeAgent.ReceivedForwardedProperties.ValueKind.Should().Be(JsonValueKind.Object); + + JsonElement tags = fakeAgent.ReceivedForwardedProperties.GetProperty("tags"); + tags.GetArrayLength().Should().Be(3); + tags[0].GetString().Should().Be("tag1"); + + JsonElement scores = fakeAgent.ReceivedForwardedProperties.GetProperty("scores"); + scores.GetArrayLength().Should().Be(5); + scores[2].GetInt32().Should().Be(3); + } + + [Fact] + public async Task ForwardedProps_WhenEmpty_DoesNotCauseErrorsAsync() + { + // Arrange + FakeForwardedPropsAgent fakeAgent = new(); + await this.SetupTestServerAsync(fakeAgent); + + const string RequestJson = """ + { + "threadId": "thread-123", + "runId": "run-456", + "messages": [{ "id": "msg-1", "role": "user", "content": "test empty props" }], + "forwardedProps": {} + } + """; + + using StringContent content = new(RequestJson, Encoding.UTF8, "application/json"); + + // Act + HttpResponseMessage response = await this._client!.PostAsync(new Uri("/agent", UriKind.Relative), content); + + // Assert + response.IsSuccessStatusCode.Should().BeTrue(); + } + + [Fact] + public async Task ForwardedProps_WhenNotProvided_AgentStillWorksAsync() + { + // Arrange + FakeForwardedPropsAgent fakeAgent = new(); + await this.SetupTestServerAsync(fakeAgent); + + const string RequestJson = """ + { + "threadId": "thread-123", + "runId": "run-456", + "messages": [{ "id": "msg-1", "role": "user", "content": "test no props" }] + } + """; + + using StringContent content = new(RequestJson, Encoding.UTF8, "application/json"); + + // Act + HttpResponseMessage response = await this._client!.PostAsync(new Uri("/agent", UriKind.Relative), content); + + // Assert + response.IsSuccessStatusCode.Should().BeTrue(); + fakeAgent.ReceivedForwardedProperties.ValueKind.Should().Be(JsonValueKind.Undefined); + } + + [Fact] + public async Task ForwardedProps_ReturnsValidSSEResponse_WithTextDeltaEventsAsync() + { + // Arrange + FakeForwardedPropsAgent fakeAgent = new(); + await this.SetupTestServerAsync(fakeAgent); + + const string RequestJson = """ + { + "threadId": "thread-123", + "runId": "run-456", + "messages": [{ "id": "msg-1", "role": "user", "content": "test response" }], + "forwardedProps": { "customProp": "value" } + } + """; + + using StringContent content = new(RequestJson, Encoding.UTF8, "application/json"); + + // Act + HttpResponseMessage response = await this._client!.PostAsync(new Uri("/agent", UriKind.Relative), content); + response.EnsureSuccessStatusCode(); + + Stream stream = await response.Content.ReadAsStreamAsync(); + List> events = []; + await foreach (SseItem item in SseParser.Create(stream).EnumerateAsync()) + { + events.Add(item); + } + + // Assert + events.Should().NotBeEmpty(); + + // SSE events have EventType = "message" and the actual type is in the JSON data + // Should have run_started event + events.Should().Contain(e => e.Data != null && e.Data.Contains("\"type\":\"RUN_STARTED\"")); + + // Should have text_message_start event + events.Should().Contain(e => e.Data != null && e.Data.Contains("\"type\":\"TEXT_MESSAGE_START\"")); + + // Should have text_message_content event with the response text + events.Should().Contain(e => e.Data != null && e.Data.Contains("\"type\":\"TEXT_MESSAGE_CONTENT\"")); + + // Should have run_finished event + events.Should().Contain(e => e.Data != null && e.Data.Contains("\"type\":\"RUN_FINISHED\"")); + } + + [Fact] + public async Task ForwardedProps_WithMixedTypes_AreCorrectlyParsedAsync() + { + // Arrange + FakeForwardedPropsAgent fakeAgent = new(); + await this.SetupTestServerAsync(fakeAgent); + + const string RequestJson = """ + { + "threadId": "thread-123", + "runId": "run-456", + "messages": [{ "id": "msg-1", "role": "user", "content": "test mixed types" }], + "forwardedProps": { + "stringProp": "text", + "numberProp": 42, + "boolProp": true, + "nullProp": null, + "arrayProp": [1, "two", false], + "objectProp": { "nested": "value" } + } + } + """; + + using StringContent content = new(RequestJson, Encoding.UTF8, "application/json"); + + // Act + HttpResponseMessage response = await this._client!.PostAsync(new Uri("/agent", UriKind.Relative), content); + + // Assert + response.IsSuccessStatusCode.Should().BeTrue(); + fakeAgent.ReceivedForwardedProperties.ValueKind.Should().Be(JsonValueKind.Object); + + fakeAgent.ReceivedForwardedProperties.GetProperty("stringProp").GetString().Should().Be("text"); + fakeAgent.ReceivedForwardedProperties.GetProperty("numberProp").GetInt32().Should().Be(42); + fakeAgent.ReceivedForwardedProperties.GetProperty("boolProp").GetBoolean().Should().BeTrue(); + fakeAgent.ReceivedForwardedProperties.GetProperty("nullProp").ValueKind.Should().Be(JsonValueKind.Null); + fakeAgent.ReceivedForwardedProperties.GetProperty("arrayProp").GetArrayLength().Should().Be(3); + fakeAgent.ReceivedForwardedProperties.GetProperty("objectProp").GetProperty("nested").GetString().Should().Be("value"); + } + + private async Task SetupTestServerAsync(FakeForwardedPropsAgent fakeAgent) + { + WebApplicationBuilder builder = WebApplication.CreateBuilder(); + builder.Services.AddAGUI(); + builder.WebHost.UseTestServer(); + + this._app = builder.Build(); + + this._app.MapAGUI("/agent", fakeAgent); + + await this._app.StartAsync(); + + TestServer testServer = this._app.Services.GetRequiredService() as TestServer + ?? throw new InvalidOperationException("TestServer not found"); + + this._client = testServer.CreateClient(); + } + + public async ValueTask DisposeAsync() + { + this._client?.Dispose(); + if (this._app != null) + { + await this._app.DisposeAsync(); + } + } +} + +[SuppressMessage("Performance", "CA1812:Avoid uninstantiated internal classes", Justification = "Instantiated in tests")] +internal sealed class FakeForwardedPropsAgent : AIAgent +{ + public FakeForwardedPropsAgent() + { + } + + public override string? Description => "Agent for forwarded properties testing"; + + public JsonElement ReceivedForwardedProperties { get; private set; } + + public override Task RunAsync(IEnumerable messages, AgentThread? thread = null, AgentRunOptions? options = null, CancellationToken cancellationToken = default) + { + return this.RunStreamingAsync(messages, thread, options, cancellationToken).ToAgentRunResponseAsync(cancellationToken); + } + + public override async IAsyncEnumerable RunStreamingAsync( + IEnumerable messages, + AgentThread? thread = null, + AgentRunOptions? options = null, + [EnumeratorCancellation] CancellationToken cancellationToken = default) + { + // Extract forwarded properties from ChatOptions.AdditionalProperties (set by AG-UI hosting layer) + if (options is ChatClientAgentRunOptions { ChatOptions.AdditionalProperties: { } properties } && + properties.TryGetValue("ag_ui_forwarded_properties", out object? propsObj) && + propsObj is JsonElement forwardedProps) + { + this.ReceivedForwardedProperties = forwardedProps; + } + + // Always return a text response + string messageId = Guid.NewGuid().ToString("N"); + yield return new AgentRunResponseUpdate + { + MessageId = messageId, + Role = ChatRole.Assistant, + Contents = [new TextContent("Forwarded props processed")] + }; + + await Task.CompletedTask; + } + + public override AgentThread GetNewThread() => new FakeInMemoryAgentThread(); + + public override AgentThread DeserializeThread(JsonElement serializedThread, JsonSerializerOptions? jsonSerializerOptions = null) + { + return new FakeInMemoryAgentThread(serializedThread, jsonSerializerOptions); + } + + private sealed class FakeInMemoryAgentThread : InMemoryAgentThread + { + public FakeInMemoryAgentThread() + : base() + { + } + + public FakeInMemoryAgentThread(JsonElement serializedThread, JsonSerializerOptions? jsonSerializerOptions = null) + : base(serializedThread, jsonSerializerOptions) + { + } + } + + public override object? GetService(Type serviceType, object? serviceKey = null) => null; +} diff --git a/dotnet/tests/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.IntegrationTests/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.IntegrationTests.csproj b/dotnet/tests/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.IntegrationTests/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.IntegrationTests.csproj index 61e65fbf59..6b909fd4f2 100644 --- a/dotnet/tests/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.IntegrationTests/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.IntegrationTests.csproj +++ b/dotnet/tests/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.IntegrationTests/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.IntegrationTests.csproj @@ -1,8 +1,7 @@ - $(ProjectsCoreTargetFrameworks) - $(ProjectsDebugCoreTargetFrameworks) + $(TargetFrameworksCore) @@ -11,21 +10,21 @@ - - - - - - + + + + + - + + diff --git a/dotnet/tests/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.IntegrationTests/SharedStateTests.cs b/dotnet/tests/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.IntegrationTests/SharedStateTests.cs new file mode 100644 index 0000000000..c96f2d92d0 --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.IntegrationTests/SharedStateTests.cs @@ -0,0 +1,441 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Collections.Generic; +using System.Diagnostics.CodeAnalysis; +using System.Linq; +using System.Net.Http; +using System.Runtime.CompilerServices; +using System.Text.Json; +using System.Threading; +using System.Threading.Tasks; +using FluentAssertions; +using Microsoft.Agents.AI.AGUI; +using Microsoft.AspNetCore.Builder; +using Microsoft.AspNetCore.Hosting.Server; +using Microsoft.AspNetCore.TestHost; +using Microsoft.Extensions.AI; +using Microsoft.Extensions.DependencyInjection; + +namespace Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.IntegrationTests; + +public sealed class SharedStateTests : IAsyncDisposable +{ + private WebApplication? _app; + private HttpClient? _client; + + [Fact] + public async Task StateSnapshot_IsReturnedAsDataContent_WithCorrectMediaTypeAsync() + { + // Arrange + var initialState = new { counter = 42, status = "active" }; + var fakeAgent = new FakeStateAgent(); + + await this.SetupTestServerAsync(fakeAgent); + var chatClient = new AGUIChatClient(this._client!, "", null); + AIAgent agent = chatClient.CreateAIAgent(instructions: null, name: "assistant", description: "Sample assistant", tools: []); + ChatClientAgentThread thread = (ChatClientAgentThread)agent.GetNewThread(); + + string stateJson = JsonSerializer.Serialize(initialState); + byte[] stateBytes = System.Text.Encoding.UTF8.GetBytes(stateJson); + DataContent stateContent = new(stateBytes, "application/json"); + ChatMessage stateMessage = new(ChatRole.System, [stateContent]); + ChatMessage userMessage = new(ChatRole.User, "update state"); + + List updates = []; + + // Act + await foreach (AgentRunResponseUpdate update in agent.RunStreamingAsync([userMessage, stateMessage], thread, new AgentRunOptions(), CancellationToken.None)) + { + updates.Add(update); + } + + // Assert + updates.Should().NotBeEmpty(); + + // Should receive state snapshot as DataContent with application/json media type + AgentRunResponseUpdate? stateUpdate = updates.FirstOrDefault(u => u.Contents.Any(c => c is DataContent dc && dc.MediaType == "application/json")); + stateUpdate.Should().NotBeNull("should receive state snapshot update"); + + DataContent? dataContent = stateUpdate!.Contents.OfType().FirstOrDefault(dc => dc.MediaType == "application/json"); + dataContent.Should().NotBeNull(); + + // Verify the state content + string receivedJson = System.Text.Encoding.UTF8.GetString(dataContent!.Data.ToArray()); + JsonElement receivedState = JsonElement.Parse(receivedJson); + receivedState.GetProperty("counter").GetInt32().Should().Be(43, "state should be incremented"); + receivedState.GetProperty("status").GetString().Should().Be("active"); + } + + [Fact] + public async Task StateSnapshot_HasCorrectAdditionalPropertiesAsync() + { + // Arrange + var initialState = new { step = 1 }; + var fakeAgent = new FakeStateAgent(); + + await this.SetupTestServerAsync(fakeAgent); + var chatClient = new AGUIChatClient(this._client!, "", null); + AIAgent agent = chatClient.CreateAIAgent(instructions: null, name: "assistant", description: "Sample assistant", tools: []); + ChatClientAgentThread thread = (ChatClientAgentThread)agent.GetNewThread(); + + string stateJson = JsonSerializer.Serialize(initialState); + byte[] stateBytes = System.Text.Encoding.UTF8.GetBytes(stateJson); + DataContent stateContent = new(stateBytes, "application/json"); + ChatMessage stateMessage = new(ChatRole.System, [stateContent]); + ChatMessage userMessage = new(ChatRole.User, "process"); + + List updates = []; + + // Act + await foreach (AgentRunResponseUpdate update in agent.RunStreamingAsync([userMessage, stateMessage], thread, new AgentRunOptions(), CancellationToken.None)) + { + updates.Add(update); + } + + // Assert + AgentRunResponseUpdate? stateUpdate = updates.FirstOrDefault(u => u.Contents.Any(c => c is DataContent dc && dc.MediaType == "application/json")); + stateUpdate.Should().NotBeNull(); + + ChatResponseUpdate chatUpdate = stateUpdate!.AsChatResponseUpdate(); + chatUpdate.AdditionalProperties.Should().NotBeNull(); + chatUpdate.AdditionalProperties.Should().ContainKey("is_state_snapshot"); + ((bool)chatUpdate.AdditionalProperties!["is_state_snapshot"]!).Should().BeTrue(); + } + + [Fact] + public async Task ComplexState_WithNestedObjectsAndArrays_RoundTripsCorrectlyAsync() + { + // Arrange + var complexState = new + { + sessionId = "test-123", + nested = new { value = "test", count = 10 }, + array = new[] { 1, 2, 3 }, + tags = new[] { "tag1", "tag2" } + }; + var fakeAgent = new FakeStateAgent(); + + await this.SetupTestServerAsync(fakeAgent); + var chatClient = new AGUIChatClient(this._client!, "", null); + AIAgent agent = chatClient.CreateAIAgent(instructions: null, name: "assistant", description: "Sample assistant", tools: []); + ChatClientAgentThread thread = (ChatClientAgentThread)agent.GetNewThread(); + + string stateJson = JsonSerializer.Serialize(complexState); + byte[] stateBytes = System.Text.Encoding.UTF8.GetBytes(stateJson); + DataContent stateContent = new(stateBytes, "application/json"); + ChatMessage stateMessage = new(ChatRole.System, [stateContent]); + ChatMessage userMessage = new(ChatRole.User, "process complex state"); + + List updates = []; + + // Act + await foreach (AgentRunResponseUpdate update in agent.RunStreamingAsync([userMessage, stateMessage], thread, new AgentRunOptions(), CancellationToken.None)) + { + updates.Add(update); + } + + // Assert + AgentRunResponseUpdate? stateUpdate = updates.FirstOrDefault(u => u.Contents.Any(c => c is DataContent dc && dc.MediaType == "application/json")); + stateUpdate.Should().NotBeNull(); + + DataContent? dataContent = stateUpdate!.Contents.OfType().FirstOrDefault(dc => dc.MediaType == "application/json"); + string receivedJson = System.Text.Encoding.UTF8.GetString(dataContent!.Data.ToArray()); + JsonElement receivedState = JsonElement.Parse(receivedJson); + + receivedState.GetProperty("sessionId").GetString().Should().Be("test-123"); + receivedState.GetProperty("nested").GetProperty("count").GetInt32().Should().Be(10); + receivedState.GetProperty("array").GetArrayLength().Should().Be(3); + receivedState.GetProperty("tags").GetArrayLength().Should().Be(2); + } + + [Fact] + public async Task StateSnapshot_CanBeUsedInSubsequentRequest_ForStateRoundTripAsync() + { + // Arrange + var initialState = new { counter = 1, sessionId = "round-trip-test" }; + var fakeAgent = new FakeStateAgent(); + + await this.SetupTestServerAsync(fakeAgent); + var chatClient = new AGUIChatClient(this._client!, "", null); + AIAgent agent = chatClient.CreateAIAgent(instructions: null, name: "assistant", description: "Sample assistant", tools: []); + ChatClientAgentThread thread = (ChatClientAgentThread)agent.GetNewThread(); + + string stateJson = JsonSerializer.Serialize(initialState); + byte[] stateBytes = System.Text.Encoding.UTF8.GetBytes(stateJson); + DataContent stateContent = new(stateBytes, "application/json"); + ChatMessage stateMessage = new(ChatRole.System, [stateContent]); + ChatMessage userMessage = new(ChatRole.User, "increment"); + + List firstRoundUpdates = []; + + // Act - First round + await foreach (AgentRunResponseUpdate update in agent.RunStreamingAsync([userMessage, stateMessage], thread, new AgentRunOptions(), CancellationToken.None)) + { + firstRoundUpdates.Add(update); + } + + // Extract state snapshot from first round + AgentRunResponseUpdate? firstStateUpdate = firstRoundUpdates.FirstOrDefault(u => u.Contents.Any(c => c is DataContent dc && dc.MediaType == "application/json")); + firstStateUpdate.Should().NotBeNull(); + DataContent? firstStateContent = firstStateUpdate!.Contents.OfType().FirstOrDefault(dc => dc.MediaType == "application/json"); + + // Second round - use returned state + ChatMessage secondStateMessage = new(ChatRole.System, [firstStateContent!]); + ChatMessage secondUserMessage = new(ChatRole.User, "increment again"); + + List secondRoundUpdates = []; + await foreach (AgentRunResponseUpdate update in agent.RunStreamingAsync([secondUserMessage, secondStateMessage], thread, new AgentRunOptions(), CancellationToken.None)) + { + secondRoundUpdates.Add(update); + } + + // Assert - Second round should have incremented counter again + AgentRunResponseUpdate? secondStateUpdate = secondRoundUpdates.FirstOrDefault(u => u.Contents.Any(c => c is DataContent dc && dc.MediaType == "application/json")); + secondStateUpdate.Should().NotBeNull(); + + DataContent? secondStateContent = secondStateUpdate!.Contents.OfType().FirstOrDefault(dc => dc.MediaType == "application/json"); + string secondStateJson = System.Text.Encoding.UTF8.GetString(secondStateContent!.Data.ToArray()); + JsonElement secondState = JsonElement.Parse(secondStateJson); + + secondState.GetProperty("counter").GetInt32().Should().Be(3, "counter should be incremented twice: 1 -> 2 -> 3"); + } + + [Fact] + public async Task WithoutState_AgentBehavesNormally_NoStateSnapshotReturnedAsync() + { + // Arrange + var fakeAgent = new FakeStateAgent(); + + await this.SetupTestServerAsync(fakeAgent); + var chatClient = new AGUIChatClient(this._client!, "", null); + AIAgent agent = chatClient.CreateAIAgent(instructions: null, name: "assistant", description: "Sample assistant", tools: []); + ChatClientAgentThread thread = (ChatClientAgentThread)agent.GetNewThread(); + + ChatMessage userMessage = new(ChatRole.User, "hello"); + + List updates = []; + + // Act + await foreach (AgentRunResponseUpdate update in agent.RunStreamingAsync([userMessage], thread, new AgentRunOptions(), CancellationToken.None)) + { + updates.Add(update); + } + + // Assert + updates.Should().NotBeEmpty(); + + // Should NOT have state snapshot when no state is sent + bool hasStateSnapshot = updates.Any(u => u.Contents.Any(c => c is DataContent dc && dc.MediaType == "application/json")); + hasStateSnapshot.Should().BeFalse("should not return state snapshot when no state is provided"); + + // Should have normal text response + updates.Should().Contain(u => u.Contents.Any(c => c is TextContent)); + } + + [Fact] + public async Task EmptyState_DoesNotTriggerStateHandlingAsync() + { + // Arrange + var emptyState = new { }; + var fakeAgent = new FakeStateAgent(); + + await this.SetupTestServerAsync(fakeAgent); + var chatClient = new AGUIChatClient(this._client!, "", null); + AIAgent agent = chatClient.CreateAIAgent(instructions: null, name: "assistant", description: "Sample assistant", tools: []); + ChatClientAgentThread thread = (ChatClientAgentThread)agent.GetNewThread(); + + string stateJson = JsonSerializer.Serialize(emptyState); + byte[] stateBytes = System.Text.Encoding.UTF8.GetBytes(stateJson); + DataContent stateContent = new(stateBytes, "application/json"); + ChatMessage stateMessage = new(ChatRole.System, [stateContent]); + ChatMessage userMessage = new(ChatRole.User, "hello"); + + List updates = []; + + // Act + await foreach (AgentRunResponseUpdate update in agent.RunStreamingAsync([userMessage, stateMessage], thread, new AgentRunOptions(), CancellationToken.None)) + { + updates.Add(update); + } + + // Assert + updates.Should().NotBeEmpty(); + + // Empty state {} should not trigger state snapshot mechanism + bool hasEmptyStateSnapshot = updates.Any(u => u.Contents.Any(c => c is DataContent dc && dc.MediaType == "application/json")); + hasEmptyStateSnapshot.Should().BeFalse("empty state should be treated as no state"); + + // Should have normal response + updates.Should().Contain(u => u.Contents.Any(c => c is TextContent)); + } + + [Fact] + public async Task NonStreamingRunAsync_WithState_ReturnsStateInResponseAsync() + { + // Arrange + var initialState = new { counter = 5 }; + var fakeAgent = new FakeStateAgent(); + + await this.SetupTestServerAsync(fakeAgent); + var chatClient = new AGUIChatClient(this._client!, "", null); + AIAgent agent = chatClient.CreateAIAgent(instructions: null, name: "assistant", description: "Sample assistant", tools: []); + ChatClientAgentThread thread = (ChatClientAgentThread)agent.GetNewThread(); + + string stateJson = JsonSerializer.Serialize(initialState); + byte[] stateBytes = System.Text.Encoding.UTF8.GetBytes(stateJson); + DataContent stateContent = new(stateBytes, "application/json"); + ChatMessage stateMessage = new(ChatRole.System, [stateContent]); + ChatMessage userMessage = new(ChatRole.User, "process"); + + // Act + AgentRunResponse response = await agent.RunAsync([userMessage, stateMessage], thread, new AgentRunOptions(), CancellationToken.None); + + // Assert + response.Should().NotBeNull(); + response.Messages.Should().NotBeEmpty(); + + // Should have message with DataContent containing state + bool hasStateMessage = response.Messages.Any(m => m.Contents.Any(c => c is DataContent dc && dc.MediaType == "application/json")); + hasStateMessage.Should().BeTrue("response should contain state message"); + + ChatMessage? stateResponseMessage = response.Messages.FirstOrDefault(m => m.Contents.Any(c => c is DataContent dc && dc.MediaType == "application/json")); + stateResponseMessage.Should().NotBeNull(); + + DataContent? dataContent = stateResponseMessage!.Contents.OfType().FirstOrDefault(dc => dc.MediaType == "application/json"); + string receivedJson = System.Text.Encoding.UTF8.GetString(dataContent!.Data.ToArray()); + JsonElement receivedState = JsonElement.Parse(receivedJson); + receivedState.GetProperty("counter").GetInt32().Should().Be(6); + } + + private async Task SetupTestServerAsync(FakeStateAgent fakeAgent) + { + WebApplicationBuilder builder = WebApplication.CreateBuilder(); + builder.Services.AddAGUI(); + builder.WebHost.UseTestServer(); + + this._app = builder.Build(); + + this._app.MapAGUI("/agent", fakeAgent); + + await this._app.StartAsync(); + + TestServer testServer = this._app.Services.GetRequiredService() as TestServer + ?? throw new InvalidOperationException("TestServer not found"); + + this._client = testServer.CreateClient(); + this._client.BaseAddress = new Uri("http://localhost/agent"); + } + + public async ValueTask DisposeAsync() + { + this._client?.Dispose(); + if (this._app != null) + { + await this._app.DisposeAsync(); + } + } +} + +[SuppressMessage("Performance", "CA1812:Avoid uninstantiated internal classes", Justification = "Instantiated in tests")] +internal sealed class FakeStateAgent : AIAgent +{ + public override string? Description => "Agent for state testing"; + + public override Task RunAsync(IEnumerable messages, AgentThread? thread = null, AgentRunOptions? options = null, CancellationToken cancellationToken = default) + { + return this.RunStreamingAsync(messages, thread, options, cancellationToken).ToAgentRunResponseAsync(cancellationToken); + } + + public override async IAsyncEnumerable RunStreamingAsync( + IEnumerable messages, + AgentThread? thread = null, + AgentRunOptions? options = null, + [EnumeratorCancellation] CancellationToken cancellationToken = default) + { + // Check for state in ChatOptions.AdditionalProperties (set by AG-UI hosting layer) + if (options is ChatClientAgentRunOptions { ChatOptions.AdditionalProperties: { } properties } && + properties.TryGetValue("ag_ui_state", out object? stateObj) && + stateObj is JsonElement state && + state.ValueKind == JsonValueKind.Object) + { + // Check if state object has properties (not empty {}) + bool hasProperties = false; + foreach (JsonProperty _ in state.EnumerateObject()) + { + hasProperties = true; + break; + } + + if (hasProperties) + { + // State is present and non-empty - modify it and return as DataContent + Dictionary modifiedState = []; + foreach (JsonProperty prop in state.EnumerateObject()) + { + if (prop.Name == "counter" && prop.Value.ValueKind == JsonValueKind.Number) + { + modifiedState[prop.Name] = prop.Value.GetInt32() + 1; + } + else if (prop.Value.ValueKind == JsonValueKind.Number) + { + modifiedState[prop.Name] = prop.Value.GetInt32(); + } + else if (prop.Value.ValueKind == JsonValueKind.String) + { + modifiedState[prop.Name] = prop.Value.GetString(); + } + else if (prop.Value.ValueKind is JsonValueKind.Object or JsonValueKind.Array) + { + modifiedState[prop.Name] = prop.Value; + } + } + + // Return modified state as DataContent + string modifiedStateJson = JsonSerializer.Serialize(modifiedState); + byte[] modifiedStateBytes = System.Text.Encoding.UTF8.GetBytes(modifiedStateJson); + DataContent modifiedStateContent = new(modifiedStateBytes, "application/json"); + + yield return new AgentRunResponseUpdate + { + MessageId = Guid.NewGuid().ToString("N"), + Role = ChatRole.Assistant, + Contents = [modifiedStateContent] + }; + } + } + + // Always return a text response + string messageId = Guid.NewGuid().ToString("N"); + yield return new AgentRunResponseUpdate + { + MessageId = messageId, + Role = ChatRole.Assistant, + Contents = [new TextContent("State processed")] + }; + + await Task.CompletedTask; + } + + public override AgentThread GetNewThread() => new FakeInMemoryAgentThread(); + + public override AgentThread DeserializeThread(JsonElement serializedThread, JsonSerializerOptions? jsonSerializerOptions = null) + { + return new FakeInMemoryAgentThread(serializedThread, jsonSerializerOptions); + } + + private sealed class FakeInMemoryAgentThread : InMemoryAgentThread + { + public FakeInMemoryAgentThread() + : base() + { + } + + public FakeInMemoryAgentThread(JsonElement serializedThread, JsonSerializerOptions? jsonSerializerOptions = null) + : base(serializedThread, jsonSerializerOptions) + { + } + } + + public override object? GetService(Type serviceType, object? serviceKey = null) => null; +} diff --git a/dotnet/tests/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.IntegrationTests/ToolCallingTests.cs b/dotnet/tests/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.IntegrationTests/ToolCallingTests.cs new file mode 100644 index 0000000000..178ed20d73 --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.IntegrationTests/ToolCallingTests.cs @@ -0,0 +1,697 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System; +using System.Collections.Generic; +using System.Linq; +using System.Net.Http; +using System.Runtime.CompilerServices; +using System.Text.Json; +using System.Text.Json.Serialization; +using System.Threading; +using System.Threading.Tasks; +using FluentAssertions; +using Microsoft.Agents.AI.AGUI; +using Microsoft.AspNetCore.Builder; +using Microsoft.AspNetCore.Hosting.Server; +using Microsoft.AspNetCore.TestHost; +using Microsoft.Extensions.AI; +using Microsoft.Extensions.DependencyInjection; +using Xunit.Abstractions; + +namespace Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.IntegrationTests; + +public sealed class ToolCallingTests : IAsyncDisposable +{ + private WebApplication? _app; + private HttpClient? _client; + private readonly ITestOutputHelper _output; + + public ToolCallingTests(ITestOutputHelper output) + { + this._output = output; + } + + [Fact] + public async Task ServerTriggersSingleFunctionCallAsync() + { + // Arrange + int callCount = 0; + AIFunction serverTool = AIFunctionFactory.Create(() => + { + callCount++; + return "Server function result"; + }, "ServerFunction", "A function on the server"); + + await this.SetupTestServerAsync(serverTools: [serverTool]); + var chatClient = new AGUIChatClient(this._client!, "", null); + AIAgent agent = chatClient.CreateAIAgent(instructions: null, name: "assistant", description: "Test assistant", tools: []); + AgentThread thread = agent.GetNewThread(); + ChatMessage userMessage = new(ChatRole.User, "Call the server function"); + + List updates = []; + + // Act + await foreach (AgentRunResponseUpdate update in agent.RunStreamingAsync([userMessage], thread, new AgentRunOptions(), CancellationToken.None)) + { + updates.Add(update); + } + + // Assert + callCount.Should().Be(1, "server function should be called once"); + updates.Should().Contain(u => u.Contents.Any(c => c is FunctionCallContent), "should contain function call"); + updates.Should().Contain(u => u.Contents.Any(c => c is FunctionResultContent), "should contain function result"); + + var functionCallUpdates = updates.Where(u => u.Contents.Any(c => c is FunctionCallContent)).ToList(); + functionCallUpdates.Should().HaveCount(1); + + var functionResultUpdates = updates.Where(u => u.Contents.Any(c => c is FunctionResultContent)).ToList(); + functionResultUpdates.Should().HaveCount(1); + + var resultContent = functionResultUpdates[0].Contents.OfType().First(); + resultContent.Result.Should().NotBeNull(); + } + + [Fact] + public async Task ServerTriggersMultipleFunctionCallsAsync() + { + // Arrange + int getWeatherCallCount = 0; + int getTimeCallCount = 0; + + AIFunction getWeatherTool = AIFunctionFactory.Create(() => + { + getWeatherCallCount++; + return "Sunny, 75°F"; + }, "GetWeather", "Gets the current weather"); + + AIFunction getTimeTool = AIFunctionFactory.Create(() => + { + getTimeCallCount++; + return "3:45 PM"; + }, "GetTime", "Gets the current time"); + + await this.SetupTestServerAsync(serverTools: [getWeatherTool, getTimeTool]); + var chatClient = new AGUIChatClient(this._client!, "", null); + AIAgent agent = chatClient.CreateAIAgent(instructions: null, name: "assistant", description: "Test assistant", tools: []); + AgentThread thread = agent.GetNewThread(); + ChatMessage userMessage = new(ChatRole.User, "What's the weather and time?"); + + List updates = []; + + // Act + await foreach (AgentRunResponseUpdate update in agent.RunStreamingAsync([userMessage], thread, new AgentRunOptions(), CancellationToken.None)) + { + updates.Add(update); + } + + // Assert + getWeatherCallCount.Should().Be(1, "GetWeather should be called once"); + getTimeCallCount.Should().Be(1, "GetTime should be called once"); + + var functionCallUpdates = updates.Where(u => u.Contents.Any(c => c is FunctionCallContent)).ToList(); + functionCallUpdates.Should().NotBeEmpty("should contain function calls"); + + var functionCalls = updates.SelectMany(u => u.Contents.OfType()).ToList(); + functionCalls.Should().HaveCount(2, "should have 2 function calls"); + functionCalls.Should().Contain(fc => fc.Name == "GetWeather"); + functionCalls.Should().Contain(fc => fc.Name == "GetTime"); + + var functionResults = updates.SelectMany(u => u.Contents.OfType()).ToList(); + functionResults.Should().HaveCount(2, "should have 2 function results"); + } + + [Fact] + public async Task ClientTriggersSingleFunctionCallAsync() + { + // Arrange + int callCount = 0; + AIFunction clientTool = AIFunctionFactory.Create(() => + { + callCount++; + return "Client function result"; + }, "ClientFunction", "A function on the client"); + + await this.SetupTestServerAsync(); + var chatClient = new AGUIChatClient(this._client!, "", null); + AIAgent agent = chatClient.CreateAIAgent(instructions: null, name: "assistant", description: "Test assistant", tools: [clientTool]); + AgentThread thread = agent.GetNewThread(); + ChatMessage userMessage = new(ChatRole.User, "Call the client function"); + + List updates = []; + + // Act + await foreach (AgentRunResponseUpdate update in agent.RunStreamingAsync([userMessage], thread, new AgentRunOptions(), CancellationToken.None)) + { + updates.Add(update); + } + + // Assert + callCount.Should().Be(1, "client function should be called once"); + updates.Should().Contain(u => u.Contents.Any(c => c is FunctionCallContent), "should contain function call"); + updates.Should().Contain(u => u.Contents.Any(c => c is FunctionResultContent), "should contain function result"); + + var functionCallUpdates = updates.Where(u => u.Contents.Any(c => c is FunctionCallContent)).ToList(); + functionCallUpdates.Should().HaveCount(1); + + var functionResultUpdates = updates.Where(u => u.Contents.Any(c => c is FunctionResultContent)).ToList(); + functionResultUpdates.Should().HaveCount(1); + + var resultContent = functionResultUpdates[0].Contents.OfType().First(); + resultContent.Result.Should().NotBeNull(); + } + + [Fact] + public async Task ClientTriggersMultipleFunctionCallsAsync() + { + // Arrange + int calculateCallCount = 0; + int formatCallCount = 0; + + AIFunction calculateTool = AIFunctionFactory.Create((int a, int b) => + { + calculateCallCount++; + return a + b; + }, "Calculate", "Calculates sum of two numbers"); + + AIFunction formatTool = AIFunctionFactory.Create((string text) => + { + formatCallCount++; + return text.ToUpperInvariant(); + }, "FormatText", "Formats text to uppercase"); + + await this.SetupTestServerAsync(); + var chatClient = new AGUIChatClient(this._client!, "", null); + AIAgent agent = chatClient.CreateAIAgent(instructions: null, name: "assistant", description: "Test assistant", tools: [calculateTool, formatTool]); + AgentThread thread = agent.GetNewThread(); + ChatMessage userMessage = new(ChatRole.User, "Calculate 5 + 3 and format 'hello'"); + + List updates = []; + + // Act + await foreach (AgentRunResponseUpdate update in agent.RunStreamingAsync([userMessage], thread, new AgentRunOptions(), CancellationToken.None)) + { + updates.Add(update); + } + + // Assert + calculateCallCount.Should().Be(1, "Calculate should be called once"); + formatCallCount.Should().Be(1, "FormatText should be called once"); + + var functionCallUpdates = updates.Where(u => u.Contents.Any(c => c is FunctionCallContent)).ToList(); + functionCallUpdates.Should().NotBeEmpty("should contain function calls"); + + var functionCalls = updates.SelectMany(u => u.Contents.OfType()).ToList(); + functionCalls.Should().HaveCount(2, "should have 2 function calls"); + functionCalls.Should().Contain(fc => fc.Name == "Calculate"); + functionCalls.Should().Contain(fc => fc.Name == "FormatText"); + + var functionResults = updates.SelectMany(u => u.Contents.OfType()).ToList(); + functionResults.Should().HaveCount(2, "should have 2 function results"); + } + + [Fact] + public async Task ServerAndClientTriggerFunctionCallsSimultaneouslyAsync() + { + // Arrange + int serverCallCount = 0; + int clientCallCount = 0; + + AIFunction serverTool = AIFunctionFactory.Create(() => + { + System.Diagnostics.Debug.Assert(true, "Server function is being called!"); + serverCallCount++; + return "Server data"; + }, "GetServerData", "Gets data from the server"); + + AIFunction clientTool = AIFunctionFactory.Create(() => + { + System.Diagnostics.Debug.Assert(true, "Client function is being called!"); + clientCallCount++; + return "Client data"; + }, "GetClientData", "Gets data from the client"); + + await this.SetupTestServerAsync(serverTools: [serverTool]); + var chatClient = new AGUIChatClient(this._client!, "", null); + AIAgent agent = chatClient.CreateAIAgent(instructions: null, name: "assistant", description: "Test assistant", tools: [clientTool]); + AgentThread thread = agent.GetNewThread(); + ChatMessage userMessage = new(ChatRole.User, "Get both server and client data"); + + List updates = []; + + // Act + await foreach (AgentRunResponseUpdate update in agent.RunStreamingAsync([userMessage], thread, new AgentRunOptions(), CancellationToken.None)) + { + updates.Add(update); + this._output.WriteLine($"Update: {update.Contents.Count} contents"); + foreach (var content in update.Contents) + { + this._output.WriteLine($" Content: {content.GetType().Name}"); + if (content is FunctionCallContent fc) + { + this._output.WriteLine($" FunctionCall: {fc.Name}"); + } + if (content is FunctionResultContent fr) + { + this._output.WriteLine($" FunctionResult: {fr.CallId} - {fr.Result}"); + } + } + } + + // Assert + this._output.WriteLine($"serverCallCount={serverCallCount}, clientCallCount={clientCallCount}"); + + // NOTE: Current limitation - server tool execution doesn't work properly in this scenario + // The FakeChatClient generates calls for both tools, but the server's FunctionInvokingChatClient + // doesn't execute the server tool. Only the client tool gets executed by the client-side + // FunctionInvokingChatClient. This appears to be a product code issue that needs investigation. + + // For now, we verify that: + // 1. Client tool executes successfully on the client + clientCallCount.Should().Be(1, "client function should execute on client"); + + // 2. Both function calls are generated and sent + var functionCallUpdates = updates.Where(u => u.Contents.Any(c => c is FunctionCallContent)).ToList(); + functionCallUpdates.Should().NotBeEmpty("should contain function calls"); + + var functionCalls = updates.SelectMany(u => u.Contents.OfType()).ToList(); + functionCalls.Should().HaveCount(2, "should have 2 function calls"); + functionCalls.Should().Contain(fc => fc.Name == "GetServerData"); + functionCalls.Should().Contain(fc => fc.Name == "GetClientData"); + + // 3. Only client function result is present (server execution not working) + var functionResults = updates.SelectMany(u => u.Contents.OfType()).ToList(); + functionResults.Should().HaveCount(1, "only client function result is present due to current limitation"); + + // Client function should succeed + var clientResult = functionResults.FirstOrDefault(fr => + functionCalls.Any(fc => fc.Name == "GetClientData" && fc.CallId == fr.CallId)); + clientResult.Should().NotBeNull("client function call should have a result"); + clientResult!.Result?.ToString().Should().Be("Client data", "client function should execute successfully"); + } + + [Fact] + public async Task FunctionCallsPreserveCallIdAndNameAsync() + { + // Arrange + AIFunction testTool = AIFunctionFactory.Create(() => "Test result", "TestFunction", "A test function"); + + await this.SetupTestServerAsync(serverTools: [testTool]); + var chatClient = new AGUIChatClient(this._client!, "", null); + AIAgent agent = chatClient.CreateAIAgent(instructions: null, name: "assistant", description: "Test assistant", tools: []); + AgentThread thread = agent.GetNewThread(); + ChatMessage userMessage = new(ChatRole.User, "Call the test function"); + + List updates = []; + + // Act + await foreach (AgentRunResponseUpdate update in agent.RunStreamingAsync([userMessage], thread, new AgentRunOptions(), CancellationToken.None)) + { + updates.Add(update); + } + + // Assert + var functionCallContent = updates.SelectMany(u => u.Contents.OfType()).FirstOrDefault(); + functionCallContent.Should().NotBeNull(); + functionCallContent!.CallId.Should().NotBeNullOrEmpty(); + functionCallContent.Name.Should().Be("TestFunction"); + + var functionResultContent = updates.SelectMany(u => u.Contents.OfType()).FirstOrDefault(); + functionResultContent.Should().NotBeNull(); + functionResultContent!.CallId.Should().Be(functionCallContent.CallId, "result should have same call ID as the call"); + } + + [Fact] + public async Task ParallelFunctionCallsFromServerAreHandledCorrectlyAsync() + { + // Arrange + int func1CallCount = 0; + int func2CallCount = 0; + + AIFunction func1 = AIFunctionFactory.Create(() => + { + func1CallCount++; + return "Result 1"; + }, "Function1", "First function"); + + AIFunction func2 = AIFunctionFactory.Create(() => + { + func2CallCount++; + return "Result 2"; + }, "Function2", "Second function"); + + await this.SetupTestServerAsync(serverTools: [func1, func2], triggerParallelCalls: true); + var chatClient = new AGUIChatClient(this._client!, "", null); + AIAgent agent = chatClient.CreateAIAgent(instructions: null, name: "assistant", description: "Test assistant", tools: []); + AgentThread thread = agent.GetNewThread(); + ChatMessage userMessage = new(ChatRole.User, "Call both functions in parallel"); + + List updates = []; + + // Act + await foreach (AgentRunResponseUpdate update in agent.RunStreamingAsync([userMessage], thread, new AgentRunOptions(), CancellationToken.None)) + { + updates.Add(update); + } + + // Assert + func1CallCount.Should().Be(1, "Function1 should be called once"); + func2CallCount.Should().Be(1, "Function2 should be called once"); + + var functionCalls = updates.SelectMany(u => u.Contents.OfType()).ToList(); + functionCalls.Should().HaveCount(2); + functionCalls.Select(fc => fc.Name).Should().Contain(s_expectedFunctionNames); + + var functionResults = updates.SelectMany(u => u.Contents.OfType()).ToList(); + functionResults.Should().HaveCount(2); + + // Each result should match its corresponding call ID + foreach (var call in functionCalls) + { + functionResults.Should().Contain(r => r.CallId == call.CallId); + } + } + + private static readonly string[] s_expectedFunctionNames = ["Function1", "Function2"]; + + [Fact] + public async Task AGUIChatClientCombinesCustomJsonSerializerOptionsAsync() + { + // This test verifies that custom JSON contexts work correctly with AGUIChatClient by testing + // that a client-defined type can be serialized successfully using the combined options + + // Arrange + await this.SetupTestServerAsync(); + + // Client uses custom JSON context + var clientJsonOptions = new JsonSerializerOptions(); + clientJsonOptions.TypeInfoResolverChain.Add(ClientJsonContext.Default); + + _ = new AGUIChatClient(this._client!, "", null, clientJsonOptions); + + // Act - Verify that both AG-UI types and custom types can be serialized + // The AGUIChatClient should have combined AGUIJsonSerializerContext with ClientJsonContext + + // Try to serialize a custom type using the ClientJsonContext + var testResponse = new ClientForecastResponse(75, 60, "Rainy"); + var json = JsonSerializer.Serialize(testResponse, ClientJsonContext.Default.ClientForecastResponse); + + // Assert + var jsonElement = JsonElement.Parse(json); + jsonElement.GetProperty("MaxTemp").GetInt32().Should().Be(75); + jsonElement.GetProperty("MinTemp").GetInt32().Should().Be(60); + jsonElement.GetProperty("Outlook").GetString().Should().Be("Rainy"); + + this._output.WriteLine("Successfully serialized custom type: " + json); + + // The actual integration is tested by the ClientToolCallWithCustomArgumentsAsync test + // which verifies that AG-UI protocol works end-to-end with custom types + } + + [Fact] + public async Task ServerToolCallWithCustomArgumentsAsync() + { + // Arrange + int callCount = 0; + AIFunction serverTool = AIFunctionFactory.Create( + (ServerForecastRequest request) => + { + callCount++; + return new ServerForecastResponse( + Temperature: 72, + Condition: request.Location == "Seattle" ? "Rainy" : "Sunny", + Humidity: 65); + }, + "GetServerForecast", + "Gets the weather forecast from server", + ServerJsonContext.Default.Options); + + await this.SetupTestServerAsync(serverTools: [serverTool], jsonSerializerOptions: ServerJsonContext.Default.Options); + var chatClient = new AGUIChatClient(this._client!, "", null, ServerJsonContext.Default.Options); + AIAgent agent = chatClient.CreateAIAgent(instructions: null, name: "assistant", description: "Test assistant", tools: []); + AgentThread thread = agent.GetNewThread(); + ChatMessage userMessage = new(ChatRole.User, "Get server forecast for Seattle for 5 days"); + + List updates = []; + + // Act + await foreach (AgentRunResponseUpdate update in agent.RunStreamingAsync([userMessage], thread, new AgentRunOptions(), CancellationToken.None)) + { + updates.Add(update); + } + + // Assert + callCount.Should().Be(1, "server function with custom arguments should be called once"); + updates.Should().Contain(u => u.Contents.Any(c => c is FunctionCallContent), "should contain function call"); + updates.Should().Contain(u => u.Contents.Any(c => c is FunctionResultContent), "should contain function result"); + + var functionCallContent = updates.SelectMany(u => u.Contents.OfType()).FirstOrDefault(); + functionCallContent.Should().NotBeNull(); + functionCallContent!.Name.Should().Be("GetServerForecast"); + + var functionResultContent = updates.SelectMany(u => u.Contents.OfType()).FirstOrDefault(); + functionResultContent.Should().NotBeNull(); + functionResultContent!.Result.Should().NotBeNull(); + } + + [Fact] + public async Task ClientToolCallWithCustomArgumentsAsync() + { + // Arrange + int callCount = 0; + AIFunction clientTool = AIFunctionFactory.Create( + (ClientForecastRequest request) => + { + callCount++; + return new ClientForecastResponse( + MaxTemp: request.City == "Portland" ? 68 : 75, + MinTemp: 55, + Outlook: "Partly Cloudy"); + }, + "GetClientForecast", + "Gets the weather forecast from client", + ClientJsonContext.Default.Options); + + await this.SetupTestServerAsync(); + var chatClient = new AGUIChatClient(this._client!, "", null, ClientJsonContext.Default.Options); + AIAgent agent = chatClient.CreateAIAgent(instructions: null, name: "assistant", description: "Test assistant", tools: [clientTool]); + AgentThread thread = agent.GetNewThread(); + ChatMessage userMessage = new(ChatRole.User, "Get client forecast for Portland with hourly data"); + + List updates = []; + + // Act + await foreach (AgentRunResponseUpdate update in agent.RunStreamingAsync([userMessage], thread, new AgentRunOptions(), CancellationToken.None)) + { + updates.Add(update); + } + + // Assert + callCount.Should().Be(1, "client function with custom arguments should be called once"); + updates.Should().Contain(u => u.Contents.Any(c => c is FunctionCallContent), "should contain function call"); + updates.Should().Contain(u => u.Contents.Any(c => c is FunctionResultContent), "should contain function result"); + + var functionCallContent = updates.SelectMany(u => u.Contents.OfType()).FirstOrDefault(); + functionCallContent.Should().NotBeNull(); + functionCallContent!.Name.Should().Be("GetClientForecast"); + + var functionResultContent = updates.SelectMany(u => u.Contents.OfType()).FirstOrDefault(); + functionResultContent.Should().NotBeNull(); + functionResultContent!.Result.Should().NotBeNull(); + } + + private async Task SetupTestServerAsync( + IList? serverTools = null, + bool triggerParallelCalls = false, + JsonSerializerOptions? jsonSerializerOptions = null) + { + WebApplicationBuilder builder = WebApplication.CreateBuilder(); + builder.Services.AddAGUI(); + builder.WebHost.UseTestServer(); + + // Configure HTTP JSON options if custom serializer options provided + if (jsonSerializerOptions?.TypeInfoResolver != null) + { + builder.Services.ConfigureHttpJsonOptions(options => + options.SerializerOptions.TypeInfoResolverChain.Add(jsonSerializerOptions.TypeInfoResolver)); + } + + this._app = builder.Build(); + // FakeChatClient will receive options.Tools containing both server and client tools (merged by framework) + var fakeChatClient = new FakeToolCallingChatClient(triggerParallelCalls, this._output, jsonSerializerOptions: jsonSerializerOptions); + AIAgent baseAgent = fakeChatClient.CreateAIAgent(instructions: null, name: "base-agent", description: "A base agent for tool testing", tools: serverTools ?? []); + this._app.MapAGUI("/agent", baseAgent); + + await this._app.StartAsync(); + + TestServer testServer = this._app.Services.GetRequiredService() as TestServer + ?? throw new InvalidOperationException("TestServer not found"); + + this._client = testServer.CreateClient(); + this._client.BaseAddress = new Uri("http://localhost/agent"); + } + + public async ValueTask DisposeAsync() + { + this._client?.Dispose(); + if (this._app != null) + { + await this._app.DisposeAsync(); + } + } +} + +internal sealed class FakeToolCallingChatClient : IChatClient +{ + private readonly bool _triggerParallelCalls; + private readonly ITestOutputHelper? _output; + public FakeToolCallingChatClient(bool triggerParallelCalls = false, ITestOutputHelper? output = null, JsonSerializerOptions? jsonSerializerOptions = null) + { + this._triggerParallelCalls = triggerParallelCalls; + this._output = output; + } + + public ChatClientMetadata Metadata => new("fake-tool-calling-chat-client"); + + public async IAsyncEnumerable GetStreamingResponseAsync( + IEnumerable messages, + ChatOptions? options = null, + [EnumeratorCancellation] CancellationToken cancellationToken = default) + { + string messageId = Guid.NewGuid().ToString("N"); + + var messageList = messages.ToList(); + this._output?.WriteLine($"[FakeChatClient] Received {messageList.Count} messages"); + + // Check if there are function results in the messages - if so, we've already done the function call loop + var hasFunctionResults = messageList.Any(m => m.Contents.Any(c => c is FunctionResultContent)); + + if (hasFunctionResults) + { + this._output?.WriteLine("[FakeChatClient] Function results present, returning final response"); + // Function results are present, return a final response + yield return new ChatResponseUpdate + { + MessageId = messageId, + Role = ChatRole.Assistant, + Contents = [new TextContent("Function calls completed successfully")] + }; + yield break; + } + + // options?.Tools contains all tools (server + client merged by framework) + var allTools = (options?.Tools ?? []).ToList(); + this._output?.WriteLine($"[FakeChatClient] Received {allTools.Count} tools to advertise"); + + if (allTools.Count == 0) + { + // No tools available, just return a simple message + yield return new ChatResponseUpdate + { + MessageId = messageId, + Role = ChatRole.Assistant, + Contents = [new TextContent("No tools available")] + }; + yield break; + } + + // Determine which tools to call based on the scenario + var toolsToCall = new List(); + + // Check message content to determine what to call + var lastUserMessage = messageList.LastOrDefault(m => m.Role == ChatRole.User)?.Text ?? ""; + + if (this._triggerParallelCalls) + { + // Call all available tools in parallel + toolsToCall.AddRange(allTools); + } + else if (lastUserMessage.Contains("both", StringComparison.OrdinalIgnoreCase) || + lastUserMessage.Contains("all", StringComparison.OrdinalIgnoreCase)) + { + // Call all available tools + toolsToCall.AddRange(allTools); + } + else + { + // Default: call all available tools + // The fake LLM doesn't distinguish between server and client tools - it just requests them all + // The FunctionInvokingChatClient layers will handle executing what they can + toolsToCall.AddRange(allTools); + } + + // Assert: Should have tools to call + System.Diagnostics.Debug.Assert(toolsToCall.Count > 0, "Should have at least one tool to call"); + + // Generate function calls + // Server's FunctionInvokingChatClient will execute server tools + // Client tool calls will be sent back to client, and client's FunctionInvokingChatClient will execute them + this._output?.WriteLine($"[FakeChatClient] Generating {toolsToCall.Count} function calls"); + foreach (var tool in toolsToCall) + { + string callId = $"call_{Guid.NewGuid():N}"; + var functionName = tool.Name ?? "UnknownFunction"; + this._output?.WriteLine($"[FakeChatClient] Calling: {functionName} (type: {tool.GetType().Name})"); + + // Generate sample arguments based on the function signature + var arguments = GenerateArgumentsForTool(functionName); + + yield return new ChatResponseUpdate + { + MessageId = messageId, + Role = ChatRole.Assistant, + Contents = [new FunctionCallContent(callId, functionName, arguments)] + }; + + await Task.Yield(); + } + } + + private static Dictionary GenerateArgumentsForTool(string functionName) + { + // Generate sample arguments based on the function name + return functionName switch + { + "GetWeather" => new Dictionary { ["location"] = "Seattle" }, + "GetTime" => [], // No parameters + "Calculate" => new Dictionary { ["a"] = 5, ["b"] = 3 }, + "FormatText" => new Dictionary { ["text"] = "hello" }, + "GetServerData" => [], // No parameters + "GetClientData" => [], // No parameters + // For custom types, the parameter name is "request" and the value is an instance of the request type + "GetServerForecast" => new Dictionary { ["request"] = new ServerForecastRequest("Seattle", 5) }, + "GetClientForecast" => new Dictionary { ["request"] = new ClientForecastRequest("Portland", true) }, + _ => [] // Default: no parameters + }; + } + + public Task GetResponseAsync( + IEnumerable messages, + ChatOptions? options = null, + CancellationToken cancellationToken = default) + { + throw new NotImplementedException(); + } + + public void Dispose() + { + } + + public object? GetService(Type serviceType, object? serviceKey = null) => null; +} + +// Custom types and serialization contexts for testing cross-boundary serialization +public record ServerForecastRequest(string Location, int Days); +public record ServerForecastResponse(int Temperature, string Condition, int Humidity); + +public record ClientForecastRequest(string City, bool IncludeHourly); +public record ClientForecastResponse(int MaxTemp, int MinTemp, string Outlook); + +[JsonSourceGenerationOptions(WriteIndented = false)] +[JsonSerializable(typeof(ServerForecastRequest))] +[JsonSerializable(typeof(ServerForecastResponse))] +internal sealed partial class ServerJsonContext : JsonSerializerContext; + +[JsonSourceGenerationOptions(WriteIndented = false)] +[JsonSerializable(typeof(ClientForecastRequest))] +[JsonSerializable(typeof(ClientForecastResponse))] +internal sealed partial class ClientJsonContext : JsonSerializerContext; diff --git a/dotnet/tests/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.UnitTests/AGUIEndpointRouteBuilderExtensionsTests.cs b/dotnet/tests/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.UnitTests/AGUIEndpointRouteBuilderExtensionsTests.cs index 5f5b9fa4ee..3e80a58369 100644 --- a/dotnet/tests/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.UnitTests/AGUIEndpointRouteBuilderExtensionsTests.cs +++ b/dotnet/tests/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.UnitTests/AGUIEndpointRouteBuilderExtensionsTests.cs @@ -38,7 +38,7 @@ public void MapAGUIAgent_MapsEndpoint_AtSpecifiedPattern() AIAgent agent = new TestAgent(); // Act - IEndpointConventionBuilder? result = AGUIEndpointRouteBuilderExtensions.MapAGUI(endpointsMock.Object, Pattern, agent); + IEndpointConventionBuilder? result = endpointsMock.Object.MapAGUI(Pattern, agent); // Assert Assert.NotNull(result); @@ -80,8 +80,8 @@ AIAgent factory(IEnumerable messages, IEnumerable tools, IE { ThreadId = "thread1", RunId = "run1", - Messages = [new AGUIMessage { Id = "m1", Role = AGUIRoles.User, Content = "Test" }], - Context = new Dictionary { ["key1"] = "value1" } + Messages = [new AGUIUserMessage { Id = "m1", Content = "Test" }], + Context = [new AGUIContextItem { Description = "key1", Value = "value1" }] }; string json = JsonSerializer.Serialize(input, AGUIJsonSerializerContext.Default.RunAgentInput); httpContext.Request.Body = new MemoryStream(Encoding.UTF8.GetBytes(json)); @@ -109,7 +109,7 @@ public async Task MapAGUIAgent_ReturnsSSEResponseStream_WithCorrectContentTypeAs { ThreadId = "thread1", RunId = "run1", - Messages = [new AGUIMessage { Id = "m1", Role = AGUIRoles.User, Content = "Test" }] + Messages = [new AGUIUserMessage { Id = "m1", Content = "Test" }] }; string json = JsonSerializer.Serialize(input, AGUIJsonSerializerContext.Default.RunAgentInput); httpContext.Request.Body = new MemoryStream(Encoding.UTF8.GetBytes(json)); @@ -136,7 +136,7 @@ public async Task MapAGUIAgent_PassesCancellationToken_ToAgentExecutionAsync() { ThreadId = "thread1", RunId = "run1", - Messages = [new AGUIMessage { Id = "m1", Role = AGUIRoles.User, Content = "Test" }] + Messages = [new AGUIUserMessage { Id = "m1", Content = "Test" }] }; string json = JsonSerializer.Serialize(input, AGUIJsonSerializerContext.Default.RunAgentInput); httpContext.Request.Body = new MemoryStream(Encoding.UTF8.GetBytes(json)); @@ -168,8 +168,8 @@ AIAgent factory(IEnumerable messages, IEnumerable tools, IE RunId = "run1", Messages = [ - new AGUIMessage { Id = "m1", Role = AGUIRoles.User, Content = "First" }, - new AGUIMessage { Id = "m2", Role = AGUIRoles.Assistant, Content = "Second" } + new AGUIUserMessage { Id = "m1", Content = "First" }, + new AGUIAssistantMessage { Id = "m2", Content = "Second" } ] }; string json = JsonSerializer.Serialize(input, AGUIJsonSerializerContext.Default.RunAgentInput); @@ -190,6 +190,264 @@ AIAgent factory(IEnumerable messages, IEnumerable tools, IE Assert.Equal("Second", capturedMessages[1].Text); } + [Fact] + public async Task MapAGUIAgent_ProducesValidAGUIEventStream_WithRunStartAndFinishAsync() + { + // Arrange + DefaultHttpContext httpContext = new(); + RunAgentInput input = new() + { + ThreadId = "thread1", + RunId = "run1", + Messages = [new AGUIUserMessage { Id = "m1", Content = "Test" }] + }; + string json = JsonSerializer.Serialize(input, AGUIJsonSerializerContext.Default.RunAgentInput); + httpContext.Request.Body = new MemoryStream(Encoding.UTF8.GetBytes(json)); + MemoryStream responseStream = new(); + httpContext.Response.Body = responseStream; + + RequestDelegate handler = this.CreateRequestDelegate((messages, tools, context, props) => new TestAgent()); + + // Act + await handler(httpContext); + + // Assert + responseStream.Position = 0; + string responseContent = Encoding.UTF8.GetString(responseStream.ToArray()); + + List events = ParseSseEvents(responseContent); + + JsonElement runStarted = Assert.Single(events, static e => e.GetProperty("type").GetString() == AGUIEventTypes.RunStarted); + JsonElement runFinished = Assert.Single(events, static e => e.GetProperty("type").GetString() == AGUIEventTypes.RunFinished); + + Assert.Equal("thread1", runStarted.GetProperty("threadId").GetString()); + Assert.Equal("run1", runStarted.GetProperty("runId").GetString()); + Assert.Equal("thread1", runFinished.GetProperty("threadId").GetString()); + Assert.Equal("run1", runFinished.GetProperty("runId").GetString()); + } + + [Fact] + public async Task MapAGUIAgent_ProducesTextMessageEvents_InCorrectOrderAsync() + { + // Arrange + DefaultHttpContext httpContext = new(); + RunAgentInput input = new() + { + ThreadId = "thread1", + RunId = "run1", + Messages = [new AGUIUserMessage { Id = "m1", Content = "Hello" }] + }; + string json = JsonSerializer.Serialize(input, AGUIJsonSerializerContext.Default.RunAgentInput); + httpContext.Request.Body = new MemoryStream(Encoding.UTF8.GetBytes(json)); + MemoryStream responseStream = new(); + httpContext.Response.Body = responseStream; + + RequestDelegate handler = this.CreateRequestDelegate((messages, tools, context, props) => new TestAgent()); + + // Act + await handler(httpContext); + + // Assert + responseStream.Position = 0; + string responseContent = Encoding.UTF8.GetString(responseStream.ToArray()); + + List events = ParseSseEvents(responseContent); + List eventTypes = new(events.Count); + foreach (JsonElement evt in events) + { + eventTypes.Add(evt.GetProperty("type").GetString()); + } + + Assert.Contains(AGUIEventTypes.RunStarted, eventTypes); + Assert.Contains(AGUIEventTypes.TextMessageContent, eventTypes); + Assert.Contains(AGUIEventTypes.RunFinished, eventTypes); + + int runStartIndex = eventTypes.IndexOf(AGUIEventTypes.RunStarted); + int firstContentIndex = eventTypes.IndexOf(AGUIEventTypes.TextMessageContent); + int runFinishIndex = eventTypes.LastIndexOf(AGUIEventTypes.RunFinished); + + Assert.True(runStartIndex < firstContentIndex, "Run start should precede text content."); + Assert.True(firstContentIndex < runFinishIndex, "Text content should precede run finish."); + } + + [Fact] + public async Task MapAGUIAgent_EmitsTextMessageContent_WithCorrectDeltaAsync() + { + // Arrange + DefaultHttpContext httpContext = new(); + RunAgentInput input = new() + { + ThreadId = "thread1", + RunId = "run1", + Messages = [new AGUIUserMessage { Id = "m1", Content = "Test" }] + }; + string json = JsonSerializer.Serialize(input, AGUIJsonSerializerContext.Default.RunAgentInput); + httpContext.Request.Body = new MemoryStream(Encoding.UTF8.GetBytes(json)); + MemoryStream responseStream = new(); + httpContext.Response.Body = responseStream; + + RequestDelegate handler = this.CreateRequestDelegate((messages, tools, context, props) => new TestAgent()); + + // Act + await handler(httpContext); + + // Assert + responseStream.Position = 0; + string responseContent = Encoding.UTF8.GetString(responseStream.ToArray()); + + List events = ParseSseEvents(responseContent); + JsonElement textContentEvent = Assert.Single(events, static e => e.GetProperty("type").GetString() == AGUIEventTypes.TextMessageContent); + + Assert.Equal("Test response", textContentEvent.GetProperty("delta").GetString()); + } + + [Fact] + public async Task MapAGUIAgent_WithCustomAgent_ProducesExpectedStreamStructureAsync() + { + // Arrange + static AIAgent CustomAgentFactory(IEnumerable messages, IEnumerable tools, IEnumerable> context, JsonElement props) + { + return new MultiResponseAgent(); + } + + DefaultHttpContext httpContext = new(); + RunAgentInput input = new() + { + ThreadId = "custom_thread", + RunId = "custom_run", + Messages = [new AGUIUserMessage { Id = "m1", Content = "Multi" }] + }; + string json = JsonSerializer.Serialize(input, AGUIJsonSerializerContext.Default.RunAgentInput); + httpContext.Request.Body = new MemoryStream(Encoding.UTF8.GetBytes(json)); + MemoryStream responseStream = new(); + httpContext.Response.Body = responseStream; + + RequestDelegate handler = this.CreateRequestDelegate(CustomAgentFactory); + + // Act + await handler(httpContext); + + // Assert + responseStream.Position = 0; + string responseContent = Encoding.UTF8.GetString(responseStream.ToArray()); + + List events = ParseSseEvents(responseContent); + List contentEvents = []; + foreach (JsonElement evt in events) + { + if (evt.GetProperty("type").GetString() == AGUIEventTypes.TextMessageContent) + { + contentEvents.Add(evt); + } + } + + Assert.True(contentEvents.Count >= 3, $"Expected at least 3 text_message.content events, got {contentEvents.Count}"); + + List deltas = new(contentEvents.Count); + foreach (JsonElement contentEvent in contentEvents) + { + deltas.Add(contentEvent.GetProperty("delta").GetString()); + } + + Assert.Contains("First", deltas); + Assert.Contains(" part", deltas); + Assert.Contains(" of response", deltas); + } + + [Fact] + public async Task MapAGUIAgent_ProducesCorrectThreadAndRunIds_InAllEventsAsync() + { + // Arrange + DefaultHttpContext httpContext = new(); + RunAgentInput input = new() + { + ThreadId = "test_thread_123", + RunId = "test_run_456", + Messages = [new AGUIUserMessage { Id = "m1", Content = "Test" }] + }; + string json = JsonSerializer.Serialize(input, AGUIJsonSerializerContext.Default.RunAgentInput); + httpContext.Request.Body = new MemoryStream(Encoding.UTF8.GetBytes(json)); + MemoryStream responseStream = new(); + httpContext.Response.Body = responseStream; + + RequestDelegate handler = this.CreateRequestDelegate((messages, tools, context, props) => new TestAgent()); + + // Act + await handler(httpContext); + + // Assert + responseStream.Position = 0; + string responseContent = Encoding.UTF8.GetString(responseStream.ToArray()); + + List events = ParseSseEvents(responseContent); + JsonElement runStarted = Assert.Single(events, static e => e.GetProperty("type").GetString() == AGUIEventTypes.RunStarted); + + Assert.Equal("test_thread_123", runStarted.GetProperty("threadId").GetString()); + Assert.Equal("test_run_456", runStarted.GetProperty("runId").GetString()); + } + + private static List ParseSseEvents(string responseContent) + { + List events = []; + using StringReader reader = new(responseContent); + StringBuilder dataBuilder = new(); + string? line; + + while ((line = reader.ReadLine()) != null) + { + if (line.StartsWith("data:", StringComparison.Ordinal)) + { + string payload = line.Length > 5 && line[5] == ' ' + ? line.Substring(6) + : line.Substring(5); + dataBuilder.Append(payload); + } + else if (line.Length == 0 && dataBuilder.Length > 0) + { + using JsonDocument document = JsonDocument.Parse(dataBuilder.ToString()); + events.Add(document.RootElement.Clone()); + dataBuilder.Clear(); + } + } + + if (dataBuilder.Length > 0) + { + using JsonDocument document = JsonDocument.Parse(dataBuilder.ToString()); + events.Add(document.RootElement.Clone()); + } + + return events; + } + + private sealed class MultiResponseAgent : AIAgent + { + protected override string? IdCore => "multi-response-agent"; + + public override string? Description => "Agent that produces multiple text chunks"; + + public override AgentThread GetNewThread() => new TestInMemoryAgentThread(); + + public override AgentThread DeserializeThread(JsonElement serializedThread, JsonSerializerOptions? jsonSerializerOptions = null) => + new TestInMemoryAgentThread(serializedThread, jsonSerializerOptions); + + public override Task RunAsync(IEnumerable messages, AgentThread? thread = null, AgentRunOptions? options = null, CancellationToken cancellationToken = default) + { + throw new NotImplementedException(); + } + + public override async IAsyncEnumerable RunStreamingAsync( + IEnumerable messages, + AgentThread? thread = null, + AgentRunOptions? options = null, + [System.Runtime.CompilerServices.EnumeratorCancellation] CancellationToken cancellationToken = default) + { + await Task.CompletedTask; + yield return new AgentRunResponseUpdate(new ChatResponseUpdate(ChatRole.Assistant, "First")); + yield return new AgentRunResponseUpdate(new ChatResponseUpdate(ChatRole.Assistant, " part")); + yield return new AgentRunResponseUpdate(new ChatResponseUpdate(ChatRole.Assistant, " of response")); + } + } + private RequestDelegate CreateRequestDelegate( Func, IEnumerable, IEnumerable>, JsonElement, AIAgent> factory) { @@ -217,17 +475,19 @@ private RequestDelegate CreateRequestDelegate( return; } - IEnumerable messages = input.Messages.AsChatMessages(); - IEnumerable> contextValues = input.Context; + IEnumerable messages = input.Messages.AsChatMessages(AGUIJsonSerializerContext.Default.Options); + IEnumerable> contextValues = input.Context.Select(c => new KeyValuePair(c.Description, c.Value)); JsonElement forwardedProps = input.ForwardedProperties; AIAgent agent = factory(messages, [], contextValues, forwardedProps); IAsyncEnumerable events = agent.RunStreamingAsync( messages, cancellationToken: cancellationToken) + .AsChatResponseUpdatesAsync() .AsAGUIEventStreamAsync( input.ThreadId, input.RunId, + AGUIJsonSerializerContext.Default.Options, cancellationToken); ILogger logger = NullLogger.Instance; @@ -250,7 +510,7 @@ public TestInMemoryAgentThread(JsonElement serializedThreadState, JsonSerializer private sealed class TestAgent : AIAgent { - public override string Id => "test-agent"; + protected override string? IdCore => "test-agent"; public override string? Description => "Test agent"; diff --git a/dotnet/tests/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.UnitTests/AGUIServerSentEventsResultTests.cs b/dotnet/tests/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.UnitTests/AGUIServerSentEventsResultTests.cs index c2a8fa9998..f049218473 100644 --- a/dotnet/tests/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.UnitTests/AGUIServerSentEventsResultTests.cs +++ b/dotnet/tests/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.UnitTests/AGUIServerSentEventsResultTests.cs @@ -100,9 +100,6 @@ public async Task ExecuteAsync_WithEmptyEventStream_CompletesSuccessfullyAsync() // Act await result.ExecuteAsync(httpContext); - - // Assert - Assert.Equal(StatusCodes.Status200OK, result.StatusCode); } [Fact] diff --git a/dotnet/tests/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.UnitTests/AgentRunResponseUpdateAGUIExtensionsTests.cs b/dotnet/tests/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.UnitTests/AgentRunResponseUpdateAGUIExtensionsTests.cs deleted file mode 100644 index 4ecd3fbe79..0000000000 --- a/dotnet/tests/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.UnitTests/AgentRunResponseUpdateAGUIExtensionsTests.cs +++ /dev/null @@ -1,165 +0,0 @@ -// Copyright (c) Microsoft. All rights reserved. - -using System.Collections.Generic; -using System.Linq; -using System.Threading; -using System.Threading.Tasks; -using Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.Shared; -using Microsoft.Extensions.AI; - -namespace Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.UnitTests; - -public sealed class AgentRunResponseUpdateAGUIExtensionsTests -{ - [Fact] - public async Task AsAGUIEventStreamAsync_YieldsRunStartedEvent_AtBeginningWithCorrectIdsAsync() - { - // Arrange - const string ThreadId = "thread1"; - const string RunId = "run1"; - List updates = []; - - // Act - List events = []; - await foreach (BaseEvent evt in updates.ToAsyncEnumerableAsync().AsAGUIEventStreamAsync(ThreadId, RunId, CancellationToken.None)) - { - events.Add(evt); - } - - // Assert - Assert.NotEmpty(events); - RunStartedEvent startEvent = Assert.IsType(events.First()); - Assert.Equal(ThreadId, startEvent.ThreadId); - Assert.Equal(RunId, startEvent.RunId); - Assert.Equal(AGUIEventTypes.RunStarted, startEvent.Type); - } - - [Fact] - public async Task AsAGUIEventStreamAsync_YieldsRunFinishedEvent_AtEndWithCorrectIdsAsync() - { - // Arrange - const string ThreadId = "thread1"; - const string RunId = "run1"; - List updates = []; - - // Act - List events = []; - await foreach (BaseEvent evt in updates.ToAsyncEnumerableAsync().AsAGUIEventStreamAsync(ThreadId, RunId, CancellationToken.None)) - { - events.Add(evt); - } - - // Assert - Assert.NotEmpty(events); - RunFinishedEvent finishEvent = Assert.IsType(events.Last()); - Assert.Equal(ThreadId, finishEvent.ThreadId); - Assert.Equal(RunId, finishEvent.RunId); - Assert.Equal(AGUIEventTypes.RunFinished, finishEvent.Type); - } - - [Fact] - public async Task AsAGUIEventStreamAsync_ConvertsTextContentUpdates_ToTextMessageEventsAsync() - { - // Arrange - const string ThreadId = "thread1"; - const string RunId = "run1"; - List updates = - [ - new AgentRunResponseUpdate(new ChatResponseUpdate(ChatRole.Assistant, "Hello") { MessageId = "msg1" }), - new AgentRunResponseUpdate(new ChatResponseUpdate(ChatRole.Assistant, " World") { MessageId = "msg1" }) - ]; - - // Act - List events = []; - await foreach (BaseEvent evt in updates.ToAsyncEnumerableAsync().AsAGUIEventStreamAsync(ThreadId, RunId, CancellationToken.None)) - { - events.Add(evt); - } - - // Assert - Assert.Contains(events, e => e is TextMessageStartEvent); - Assert.Contains(events, e => e is TextMessageContentEvent); - Assert.Contains(events, e => e is TextMessageEndEvent); - } - - [Fact] - public async Task AsAGUIEventStreamAsync_GroupsConsecutiveUpdates_WithSameMessageIdAsync() - { - // Arrange - const string ThreadId = "thread1"; - const string RunId = "run1"; - const string MessageId = "msg1"; - List updates = - [ - new AgentRunResponseUpdate(new ChatResponseUpdate(ChatRole.Assistant, "Hello") { MessageId = MessageId }), - new AgentRunResponseUpdate(new ChatResponseUpdate(ChatRole.Assistant, " ") { MessageId = MessageId }), - new AgentRunResponseUpdate(new ChatResponseUpdate(ChatRole.Assistant, "World") { MessageId = MessageId }) - ]; - - // Act - List events = []; - await foreach (BaseEvent evt in updates.ToAsyncEnumerableAsync().AsAGUIEventStreamAsync(ThreadId, RunId, CancellationToken.None)) - { - events.Add(evt); - } - - // Assert - List startEvents = events.OfType().ToList(); - List endEvents = events.OfType().ToList(); - Assert.Single(startEvents); - Assert.Single(endEvents); - Assert.Equal(MessageId, startEvents[0].MessageId); - Assert.Equal(MessageId, endEvents[0].MessageId); - } - - [Fact] - public async Task AsAGUIEventStreamAsync_WithRoleChanges_EmitsProperTextMessageStartEventsAsync() - { - // Arrange - const string ThreadId = "thread1"; - const string RunId = "run1"; - List updates = - [ - new AgentRunResponseUpdate(new ChatResponseUpdate(ChatRole.Assistant, "Hello") { MessageId = "msg1" }), - new AgentRunResponseUpdate(new ChatResponseUpdate(ChatRole.User, "Hi") { MessageId = "msg2" }) - ]; - - // Act - List events = []; - await foreach (BaseEvent evt in updates.ToAsyncEnumerableAsync().AsAGUIEventStreamAsync(ThreadId, RunId, CancellationToken.None)) - { - events.Add(evt); - } - - // Assert - List startEvents = events.OfType().ToList(); - Assert.Equal(2, startEvents.Count); - Assert.Equal("msg1", startEvents[0].MessageId); - Assert.Equal("msg2", startEvents[1].MessageId); - } - - [Fact] - public async Task AsAGUIEventStreamAsync_EmitsTextMessageEndEvent_WhenMessageIdChangesAsync() - { - // Arrange - const string ThreadId = "thread1"; - const string RunId = "run1"; - List updates = - [ - new AgentRunResponseUpdate(new ChatResponseUpdate(ChatRole.Assistant, "First") { MessageId = "msg1" }), - new AgentRunResponseUpdate(new ChatResponseUpdate(ChatRole.Assistant, "Second") { MessageId = "msg2" }) - ]; - - // Act - List events = []; - await foreach (BaseEvent evt in updates.ToAsyncEnumerableAsync().AsAGUIEventStreamAsync(ThreadId, RunId, CancellationToken.None)) - { - events.Add(evt); - } - - // Assert - List endEvents = events.OfType().ToList(); - Assert.NotEmpty(endEvents); - Assert.Contains(endEvents, e => e.MessageId == "msg1"); - } -} diff --git a/dotnet/tests/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.UnitTests/ChatResponseUpdateAGUIExtensionsTests.cs b/dotnet/tests/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.UnitTests/ChatResponseUpdateAGUIExtensionsTests.cs new file mode 100644 index 0000000000..bf2aa6fb0b --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.UnitTests/ChatResponseUpdateAGUIExtensionsTests.cs @@ -0,0 +1,286 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Collections.Generic; +using System.Linq; +using System.Threading; +using System.Threading.Tasks; +using Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.Shared; +using Microsoft.Extensions.AI; + +namespace Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.UnitTests; + +public sealed class ChatResponseUpdateAGUIExtensionsTests +{ + [Fact] + public async Task AsAGUIEventStreamAsync_YieldsRunStartedEvent_AtBeginningWithCorrectIdsAsync() + { + // Arrange + const string ThreadId = "thread1"; + const string RunId = "run1"; + List updates = []; + + // Act + List events = []; + await foreach (BaseEvent evt in updates.ToAsyncEnumerableAsync().AsAGUIEventStreamAsync(ThreadId, RunId, AGUIJsonSerializerContext.Default.Options, CancellationToken.None)) + { + events.Add(evt); + } + + // Assert + Assert.NotEmpty(events); + RunStartedEvent startEvent = Assert.IsType(events.First()); + Assert.Equal(ThreadId, startEvent.ThreadId); + Assert.Equal(RunId, startEvent.RunId); + Assert.Equal(AGUIEventTypes.RunStarted, startEvent.Type); + } + + [Fact] + public async Task AsAGUIEventStreamAsync_YieldsRunFinishedEvent_AtEndWithCorrectIdsAsync() + { + // Arrange + const string ThreadId = "thread1"; + const string RunId = "run1"; + List updates = []; + + // Act + List events = []; + await foreach (BaseEvent evt in updates.ToAsyncEnumerableAsync().AsAGUIEventStreamAsync(ThreadId, RunId, AGUIJsonSerializerContext.Default.Options, CancellationToken.None)) + { + events.Add(evt); + } + + // Assert + Assert.NotEmpty(events); + RunFinishedEvent finishEvent = Assert.IsType(events.Last()); + Assert.Equal(ThreadId, finishEvent.ThreadId); + Assert.Equal(RunId, finishEvent.RunId); + Assert.Equal(AGUIEventTypes.RunFinished, finishEvent.Type); + } + + [Fact] + public async Task AsAGUIEventStreamAsync_ConvertsTextContentUpdates_ToTextMessageEventsAsync() + { + // Arrange + const string ThreadId = "thread1"; + const string RunId = "run1"; + List updates = + [ + new ChatResponseUpdate(ChatRole.Assistant, "Hello") { MessageId = "msg1" }, + new ChatResponseUpdate(ChatRole.Assistant, " World") { MessageId = "msg1" } + ]; + + // Act + List events = []; + await foreach (BaseEvent evt in updates.ToAsyncEnumerableAsync().AsAGUIEventStreamAsync(ThreadId, RunId, AGUIJsonSerializerContext.Default.Options, CancellationToken.None)) + { + events.Add(evt); + } + + // Assert + Assert.Contains(events, e => e is TextMessageStartEvent); + Assert.Contains(events, e => e is TextMessageContentEvent); + Assert.Contains(events, e => e is TextMessageEndEvent); + } + + [Fact] + public async Task AsAGUIEventStreamAsync_GroupsConsecutiveUpdates_WithSameMessageIdAsync() + { + // Arrange + const string ThreadId = "thread1"; + const string RunId = "run1"; + const string MessageId = "msg1"; + List updates = + [ + new ChatResponseUpdate(ChatRole.Assistant, "Hello") { MessageId = MessageId }, + new ChatResponseUpdate(ChatRole.Assistant, " ") { MessageId = MessageId }, + new ChatResponseUpdate(ChatRole.Assistant, "World") { MessageId = MessageId } + ]; + + // Act + List events = []; + await foreach (BaseEvent evt in updates.ToAsyncEnumerableAsync().AsAGUIEventStreamAsync(ThreadId, RunId, AGUIJsonSerializerContext.Default.Options, CancellationToken.None)) + { + events.Add(evt); + } + + // Assert + List startEvents = events.OfType().ToList(); + List endEvents = events.OfType().ToList(); + Assert.Single(startEvents); + Assert.Single(endEvents); + Assert.Equal(MessageId, startEvents[0].MessageId); + Assert.Equal(MessageId, endEvents[0].MessageId); + } + + [Fact] + public async Task AsAGUIEventStreamAsync_WithRoleChanges_EmitsProperTextMessageStartEventsAsync() + { + // Arrange + const string ThreadId = "thread1"; + const string RunId = "run1"; + List updates = + [ + new ChatResponseUpdate(ChatRole.Assistant, "Hello") { MessageId = "msg1" }, + new ChatResponseUpdate(ChatRole.User, "Hi") { MessageId = "msg2" } + ]; + + // Act + List events = []; + await foreach (BaseEvent evt in updates.ToAsyncEnumerableAsync().AsAGUIEventStreamAsync(ThreadId, RunId, AGUIJsonSerializerContext.Default.Options, CancellationToken.None)) + { + events.Add(evt); + } + + // Assert + List startEvents = events.OfType().ToList(); + Assert.Equal(2, startEvents.Count); + Assert.Equal("msg1", startEvents[0].MessageId); + Assert.Equal("msg2", startEvents[1].MessageId); + } + + [Fact] + public async Task AsAGUIEventStreamAsync_EmitsTextMessageEndEvent_WhenMessageIdChangesAsync() + { + // Arrange + const string ThreadId = "thread1"; + const string RunId = "run1"; + List updates = + [ + new ChatResponseUpdate(ChatRole.Assistant, "First") { MessageId = "msg1" }, + new ChatResponseUpdate(ChatRole.Assistant, "Second") { MessageId = "msg2" } + ]; + + // Act + List events = []; + await foreach (BaseEvent evt in updates.ToAsyncEnumerableAsync().AsAGUIEventStreamAsync(ThreadId, RunId, AGUIJsonSerializerContext.Default.Options, CancellationToken.None)) + { + events.Add(evt); + } + + // Assert + List endEvents = events.OfType().ToList(); + Assert.NotEmpty(endEvents); + Assert.Contains(endEvents, e => e.MessageId == "msg1"); + } + + [Fact] + public async Task AsAGUIEventStreamAsync_WithFunctionCallContent_EmitsToolCallEventsAsync() + { + // Arrange + const string ThreadId = "thread1"; + const string RunId = "run1"; + Dictionary arguments = new() { ["location"] = "Seattle", ["units"] = "fahrenheit" }; + FunctionCallContent functionCall = new("call_123", "GetWeather", arguments); + List updates = + [ + new ChatResponseUpdate(ChatRole.Assistant, [functionCall]) { MessageId = "msg1" } + ]; + + // Act + List events = []; + await foreach (BaseEvent evt in updates.ToAsyncEnumerableAsync().AsAGUIEventStreamAsync(ThreadId, RunId, AGUIJsonSerializerContext.Default.Options, CancellationToken.None)) + { + events.Add(evt); + } + + // Assert + ToolCallStartEvent? startEvent = events.OfType().FirstOrDefault(); + Assert.NotNull(startEvent); + Assert.Equal("call_123", startEvent.ToolCallId); + Assert.Equal("GetWeather", startEvent.ToolCallName); + Assert.Equal("msg1", startEvent.ParentMessageId); + + ToolCallArgsEvent? argsEvent = events.OfType().FirstOrDefault(); + Assert.NotNull(argsEvent); + Assert.Equal("call_123", argsEvent.ToolCallId); + Assert.Contains("location", argsEvent.Delta); + Assert.Contains("Seattle", argsEvent.Delta); + + ToolCallEndEvent? endEvent = events.OfType().FirstOrDefault(); + Assert.NotNull(endEvent); + Assert.Equal("call_123", endEvent.ToolCallId); + } + + [Fact] + public async Task AsAGUIEventStreamAsync_WithMultipleFunctionCalls_EmitsAllToolCallEventsAsync() + { + // Arrange + const string ThreadId = "thread1"; + const string RunId = "run1"; + FunctionCallContent call1 = new("call_1", "Tool1", new Dictionary()); + FunctionCallContent call2 = new("call_2", "Tool2", new Dictionary()); + ChatResponseUpdate response = new(ChatRole.Assistant, [call1, call2]) { MessageId = "msg1" }; + List updates = [response]; + + // Act + List events = []; + await foreach (BaseEvent evt in updates.ToAsyncEnumerableAsync().AsAGUIEventStreamAsync(ThreadId, RunId, AGUIJsonSerializerContext.Default.Options, CancellationToken.None)) + { + events.Add(evt); + } + + // Assert + List startEvents = events.OfType().ToList(); + Assert.Equal(2, startEvents.Count); + Assert.Contains(startEvents, e => e.ToolCallId == "call_1" && e.ToolCallName == "Tool1"); + Assert.Contains(startEvents, e => e.ToolCallId == "call_2" && e.ToolCallName == "Tool2"); + + List endEvents = events.OfType().ToList(); + Assert.Equal(2, endEvents.Count); + } + + [Fact] + public async Task AsAGUIEventStreamAsync_WithFunctionCallWithNullArguments_EmitsEventsCorrectlyAsync() + { + // Arrange + const string ThreadId = "thread1"; + const string RunId = "run1"; + FunctionCallContent functionCall = new("call_456", "NoArgsTool", null); + List updates = + [ + new ChatResponseUpdate(ChatRole.Assistant, [functionCall]) { MessageId = "msg1" } + ]; + + // Act + List events = []; + await foreach (BaseEvent evt in updates.ToAsyncEnumerableAsync().AsAGUIEventStreamAsync(ThreadId, RunId, AGUIJsonSerializerContext.Default.Options, CancellationToken.None)) + { + events.Add(evt); + } + + // Assert + Assert.Contains(events, e => e is ToolCallStartEvent); + Assert.Contains(events, e => e is ToolCallArgsEvent); + Assert.Contains(events, e => e is ToolCallEndEvent); + } + + [Fact] + public async Task AsAGUIEventStreamAsync_WithMixedContentTypes_EmitsAllEventTypesAsync() + { + // Arrange + const string ThreadId = "thread1"; + const string RunId = "run1"; + List updates = + [ + new ChatResponseUpdate(ChatRole.Assistant, "Text message") { MessageId = "msg1" }, + new ChatResponseUpdate(ChatRole.Assistant, [new FunctionCallContent("call_1", "Tool1", null)]) { MessageId = "msg2" } + ]; + + // Act + List events = []; + await foreach (BaseEvent evt in updates.ToAsyncEnumerableAsync().AsAGUIEventStreamAsync(ThreadId, RunId, AGUIJsonSerializerContext.Default.Options, CancellationToken.None)) + { + events.Add(evt); + } + + // Assert + Assert.Contains(events, e => e is RunStartedEvent); + Assert.Contains(events, e => e is TextMessageStartEvent); + Assert.Contains(events, e => e is TextMessageContentEvent); + Assert.Contains(events, e => e is TextMessageEndEvent); + Assert.Contains(events, e => e is ToolCallStartEvent); + Assert.Contains(events, e => e is ToolCallArgsEvent); + Assert.Contains(events, e => e is ToolCallEndEvent); + Assert.Contains(events, e => e is RunFinishedEvent); + } +} diff --git a/dotnet/tests/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.UnitTests/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.UnitTests.csproj b/dotnet/tests/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.UnitTests/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.UnitTests.csproj index e6d4459c6e..57a653d9f0 100644 --- a/dotnet/tests/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.UnitTests/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.UnitTests.csproj +++ b/dotnet/tests/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.UnitTests/Microsoft.Agents.AI.Hosting.AGUI.AspNetCore.UnitTests.csproj @@ -1,17 +1,17 @@ - $(ProjectsCoreTargetFrameworks) - $(ProjectsDebugCoreTargetFrameworks) + $(TargetFrameworksCore) - - - + + + + diff --git a/dotnet/tests/Microsoft.Agents.AI.Hosting.AzureFunctions.IntegrationTests/Microsoft.Agents.AI.Hosting.AzureFunctions.IntegrationTests.csproj b/dotnet/tests/Microsoft.Agents.AI.Hosting.AzureFunctions.IntegrationTests/Microsoft.Agents.AI.Hosting.AzureFunctions.IntegrationTests.csproj new file mode 100644 index 0000000000..fb955c3162 --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.Hosting.AzureFunctions.IntegrationTests/Microsoft.Agents.AI.Hosting.AzureFunctions.IntegrationTests.csproj @@ -0,0 +1,17 @@ + + + + $(TargetFrameworksCore) + enable + b7762d10-e29b-4bb1-8b74-b6d69a667dd4 + + + + + + + + + + + diff --git a/dotnet/tests/Microsoft.Agents.AI.Hosting.AzureFunctions.IntegrationTests/SamplesValidation.cs b/dotnet/tests/Microsoft.Agents.AI.Hosting.AzureFunctions.IntegrationTests/SamplesValidation.cs new file mode 100644 index 0000000000..0ba879f024 --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.Hosting.AzureFunctions.IntegrationTests/SamplesValidation.cs @@ -0,0 +1,813 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Diagnostics; +using System.Reflection; +using System.Text; +using System.Text.Json; +using Microsoft.Extensions.Configuration; +using Microsoft.Extensions.Logging; +using ModelContextProtocol.Client; +using ModelContextProtocol.Protocol; +using Xunit.Abstractions; + +namespace Microsoft.Agents.AI.Hosting.AzureFunctions.IntegrationTests; + +[Collection("Samples")] +[Trait("Category", "SampleValidation")] +public sealed class SamplesValidation(ITestOutputHelper outputHelper) : IAsyncLifetime +{ + private const string AzureFunctionsPort = "7071"; + private const string AzuritePort = "10000"; + private const string DtsPort = "8080"; + + private static readonly string s_dotnetTargetFramework = GetTargetFramework(); + private static readonly HttpClient s_sharedHttpClient = new(); + private static readonly IConfiguration s_configuration = + new ConfigurationBuilder() + .AddUserSecrets(Assembly.GetExecutingAssembly()) + .AddEnvironmentVariables() + .Build(); + + private static bool s_infrastructureStarted; + private static readonly TimeSpan s_orchestrationTimeout = TimeSpan.FromMinutes(1); + private static readonly string s_samplesPath = Path.GetFullPath( + Path.Combine(AppDomain.CurrentDomain.BaseDirectory, "..", "..", "..", "..", "..", "samples", "AzureFunctions")); + + private readonly ITestOutputHelper _outputHelper = outputHelper; + + async Task IAsyncLifetime.InitializeAsync() + { + if (!s_infrastructureStarted) + { + await this.StartSharedInfrastructureAsync(); + s_infrastructureStarted = true; + } + } + + async Task IAsyncLifetime.DisposeAsync() + { + // Nothing to clean up + await Task.CompletedTask; + } + + [Fact] + public async Task SingleAgentSampleValidationAsync() + { + string samplePath = Path.Combine(s_samplesPath, "01_SingleAgent"); + await this.RunSampleTestAsync(samplePath, async (logs) => + { + Uri startUri = new($"http://localhost:{AzureFunctionsPort}/api/agents/Joker/run"); + this._outputHelper.WriteLine($"Starting single agent orchestration via POST request to {startUri}..."); + + // Test the agent endpoint as described in the README + const string RequestBody = "Tell me a joke about a pirate."; + using HttpContent content = new StringContent(RequestBody, Encoding.UTF8, "text/plain"); + + using HttpResponseMessage response = await s_sharedHttpClient.PostAsync(startUri, content); + + // The response is expected to be a plain text response with the agent's reply (the joke) + Assert.True(response.IsSuccessStatusCode, $"Agent request failed with status: {response.StatusCode}"); + Assert.Equal("text/plain", response.Content.Headers.ContentType?.MediaType); + string responseText = await response.Content.ReadAsStringAsync(); + Assert.NotEmpty(responseText); + this._outputHelper.WriteLine($"Agent run response: {responseText}"); + + // The response headers should include the agent thread ID, which can be used to continue the conversation. + string? threadId = response.Headers.GetValues("x-ms-thread-id")?.FirstOrDefault(); + Assert.NotNull(threadId); + Assert.NotEmpty(threadId); + + this._outputHelper.WriteLine($"Agent thread ID: {threadId}"); + + // Wait for up to 30 seconds to see if the agent response is available in the logs + await this.WaitForConditionAsync( + condition: () => + { + lock (logs) + { + bool exists = logs.Any( + log => log.Message.Contains("Response:") && log.Message.Contains(threadId)); + return Task.FromResult(exists); + } + }, + message: "Agent response is available", + timeout: TimeSpan.FromSeconds(30)); + }); + } + + [Fact] + public async Task SingleAgentOrchestrationChainingSampleValidationAsync() + { + string samplePath = Path.Combine(s_samplesPath, "02_AgentOrchestration_Chaining"); + await this.RunSampleTestAsync(samplePath, async (logs) => + { + Uri startUri = new($"http://localhost:{AzureFunctionsPort}/api/singleagent/run"); + this._outputHelper.WriteLine($"Starting single agent orchestration via POST request to {startUri}..."); + + // Start the orchestration + using HttpResponseMessage startResponse = await s_sharedHttpClient.PostAsync(startUri, content: null); + + Assert.True( + startResponse.IsSuccessStatusCode, + $"Start orchestration failed with status: {startResponse.StatusCode}"); + string startResponseText = await startResponse.Content.ReadAsStringAsync(); + JsonElement startResult = JsonElement.Parse(startResponseText); + + Assert.True(startResult.TryGetProperty("statusQueryGetUri", out JsonElement statusUriElement)); + Uri statusUri = new(statusUriElement.GetString()!); + + // Wait for orchestration to complete + await this.WaitForOrchestrationCompletionAsync(statusUri); + + // Verify the final result + using HttpResponseMessage statusResponse = await s_sharedHttpClient.GetAsync(statusUri); + Assert.True( + statusResponse.IsSuccessStatusCode, + $"Status check failed with status: {statusResponse.StatusCode}"); + + string statusText = await statusResponse.Content.ReadAsStringAsync(); + JsonElement statusResult = JsonElement.Parse(statusText); + + Assert.Equal("Completed", statusResult.GetProperty("runtimeStatus").GetString()); + Assert.True(statusResult.TryGetProperty("output", out JsonElement outputElement)); + string? output = outputElement.GetString(); + + // Can't really validate the output since it's non-deterministic, but we can at least check it's non-empty + Assert.NotNull(output); + Assert.True(output.Length > 20, "Output is unexpectedly short"); + }); + } + + [Fact] + public async Task MultiAgentOrchestrationConcurrentSampleValidationAsync() + { + string samplePath = Path.Combine(s_samplesPath, "03_AgentOrchestration_Concurrency"); + await this.RunSampleTestAsync(samplePath, async (logs) => + { + // Start the multi-agent orchestration + const string RequestBody = "What is temperature?"; + using HttpContent content = new StringContent(RequestBody, Encoding.UTF8, "text/plain"); + + Uri startUri = new($"http://localhost:{AzureFunctionsPort}/api/multiagent/run"); + this._outputHelper.WriteLine($"Starting multi agent orchestration via POST request to {startUri}..."); + using HttpResponseMessage startResponse = await s_sharedHttpClient.PostAsync(startUri, content); + + Assert.True(startResponse.IsSuccessStatusCode, $"Start orchestration failed with status: {startResponse.StatusCode}"); + string startResponseText = await startResponse.Content.ReadAsStringAsync(); + JsonElement startResult = JsonElement.Parse(startResponseText); + + Assert.True(startResult.TryGetProperty("instanceId", out JsonElement instanceIdElement)); + Assert.True(startResult.TryGetProperty("statusQueryGetUri", out JsonElement statusUriElement)); + + Uri statusUri = new(statusUriElement.GetString()!); + + // Wait for orchestration to complete + await this.WaitForOrchestrationCompletionAsync(statusUri); + + // Verify the final result + using HttpResponseMessage statusResponse = await s_sharedHttpClient.GetAsync(statusUri); + Assert.True(statusResponse.IsSuccessStatusCode, $"Status check failed with status: {statusResponse.StatusCode}"); + + string statusText = await statusResponse.Content.ReadAsStringAsync(); + JsonElement statusResult = JsonElement.Parse(statusText); + + Assert.Equal("Completed", statusResult.GetProperty("runtimeStatus").GetString()); + Assert.True(statusResult.TryGetProperty("output", out JsonElement outputElement)); + + // Verify both physicist and chemist responses are present + Assert.True(outputElement.TryGetProperty("physicist", out JsonElement physicistElement)); + Assert.True(outputElement.TryGetProperty("chemist", out JsonElement chemistElement)); + + string physicistResponse = physicistElement.GetString()!; + string chemistResponse = chemistElement.GetString()!; + + Assert.NotEmpty(physicistResponse); + Assert.NotEmpty(chemistResponse); + Assert.Contains("temperature", physicistResponse, StringComparison.OrdinalIgnoreCase); + Assert.Contains("temperature", chemistResponse, StringComparison.OrdinalIgnoreCase); + }); + } + + [Fact] + public async Task MultiAgentOrchestrationConditionalsSampleValidationAsync() + { + string samplePath = Path.Combine(s_samplesPath, "04_AgentOrchestration_Conditionals"); + await this.RunSampleTestAsync(samplePath, async (logs) => + { + // Test with legitimate email + await this.TestSpamDetectionAsync("email-001", + "Hi John, I hope you're doing well. I wanted to follow up on our meeting yesterday about the quarterly report. Could you please send me the updated figures by Friday? Thanks!", + expectedSpam: false); + + // Test with spam email + await this.TestSpamDetectionAsync("email-002", + "URGENT! You've won $1,000,000! Click here now to claim your prize! Limited time offer! Don't miss out!", + expectedSpam: true); + }); + } + + [Fact] + public async Task SingleAgentOrchestrationHITLSampleValidationAsync() + { + string samplePath = Path.Combine(s_samplesPath, "05_AgentOrchestration_HITL"); + + await this.RunSampleTestAsync(samplePath, async (logs) => + { + // Start the HITL orchestration with short timeout for testing + // TODO: Add validation for the approval case + object requestBody = new + { + topic = "The Future of Artificial Intelligence", + max_review_attempts = 3, + approval_timeout_hours = 0.001 // Very short timeout for testing + }; + + string jsonContent = JsonSerializer.Serialize(requestBody); + using HttpContent content = new StringContent(jsonContent, Encoding.UTF8, "application/json"); + + Uri startUri = new($"http://localhost:{AzureFunctionsPort}/api/hitl/run"); + this._outputHelper.WriteLine($"Starting HITL orchestration via POST request to {startUri}..."); + using HttpResponseMessage startResponse = await s_sharedHttpClient.PostAsync(startUri, content); + + Assert.True( + startResponse.IsSuccessStatusCode, + $"Start HITL orchestration failed with status: {startResponse.StatusCode}"); + string startResponseText = await startResponse.Content.ReadAsStringAsync(); + JsonElement startResult = JsonElement.Parse(startResponseText); + + Assert.True(startResult.TryGetProperty("statusQueryGetUri", out JsonElement statusUriElement)); + Uri statusUri = new(statusUriElement.GetString()!); + + // Wait for orchestration to complete (it should timeout due to short timeout) + await this.WaitForOrchestrationCompletionAsync(statusUri); + + // Verify the final result + using HttpResponseMessage statusResponse = await s_sharedHttpClient.GetAsync(statusUri); + Assert.True( + statusResponse.IsSuccessStatusCode, + $"Status check failed with status: {statusResponse.StatusCode}"); + + string statusText = await statusResponse.Content.ReadAsStringAsync(); + this._outputHelper.WriteLine($"HITL orchestration status text: {statusText}"); + + JsonElement statusResult = JsonElement.Parse(statusText); + + // The orchestration should complete with a failed status due to timeout + Assert.Equal("Failed", statusResult.GetProperty("runtimeStatus").GetString()); + Assert.True(statusResult.TryGetProperty("failureDetails", out JsonElement failureDetailsElement)); + Assert.True(failureDetailsElement.TryGetProperty("ErrorType", out JsonElement errorTypeElement)); + Assert.Equal("System.TimeoutException", errorTypeElement.GetString()); + Assert.True(failureDetailsElement.TryGetProperty("ErrorMessage", out JsonElement errorMessageElement)); + Assert.StartsWith("Human approval timed out", errorMessageElement.GetString()); + }); + } + + [Fact] + public async Task LongRunningToolsSampleValidationAsync() + { + string samplePath = Path.Combine(s_samplesPath, "06_LongRunningTools"); + + await this.RunSampleTestAsync(samplePath, async (logs) => + { + // Test starting an agent that schedules a content generation orchestration + const string Prompt = "Start a content generation workflow for the topic 'The Future of Artificial Intelligence'"; + using HttpContent messageContent = new StringContent(Prompt, Encoding.UTF8, "text/plain"); + + Uri runAgentUri = new($"http://localhost:{AzureFunctionsPort}/api/agents/publisher/run"); + + this._outputHelper.WriteLine($"Starting agent tool orchestration via POST request to {runAgentUri}..."); + using HttpResponseMessage startResponse = await s_sharedHttpClient.PostAsync(runAgentUri, messageContent); + + Assert.True( + startResponse.IsSuccessStatusCode, + $"Start agent request failed with status: {startResponse.StatusCode}"); + + string startResponseText = await startResponse.Content.ReadAsStringAsync(); + this._outputHelper.WriteLine($"Agent response: {startResponseText}"); + + // The response should be deserializable as an AgentRunResponse object and have a valid thread ID + startResponse.Headers.TryGetValues("x-ms-thread-id", out IEnumerable? agentIdValues); + string? threadId = agentIdValues?.FirstOrDefault(); + Assert.NotNull(threadId); + Assert.NotEmpty(threadId); + + // Wait for the orchestration to report that it's waiting for human approval + await this.WaitForConditionAsync( + condition: () => + { + // For now, we have to rely on the logs to check for the "NOTIFICATION" message that gets generated by the activity function. + // TODO: Synchronously prompt the agent for status + lock (logs) + { + bool exists = logs.Any(log => log.Message.Contains("NOTIFICATION: Please review the following content for approval")); + return Task.FromResult(exists); + } + }, + message: "Orchestration is requesting human feedback", + timeout: TimeSpan.FromSeconds(60)); + + // Approve the content + Uri approvalUri = new($"{runAgentUri}?thread_id={threadId}"); + using HttpContent approvalContent = new StringContent("Approve the content", Encoding.UTF8, "text/plain"); + using HttpResponseMessage approvalResponse = await s_sharedHttpClient.PostAsync(approvalUri, approvalContent); + Assert.True(approvalResponse.IsSuccessStatusCode, $"Approve content request failed with status: {approvalResponse.StatusCode}"); + + // Wait for the publish notification to be logged + await this.WaitForConditionAsync( + condition: () => + { + lock (logs) + { + // TODO: Synchronously prompt the agent for status + bool exists = logs.Any(log => log.Message.Contains("PUBLISHING: Content has been published successfully")); + return Task.FromResult(exists); + } + }, + message: "Content published notification is logged", + timeout: TimeSpan.FromSeconds(60)); + + // Verify the final orchestration status by asking the agent for the status + Uri statusUri = new($"{runAgentUri}?thread_id={threadId}"); + await this.WaitForConditionAsync( + condition: async () => + { + this._outputHelper.WriteLine($"Checking status of orchestration at {statusUri}..."); + + using StringContent content = new("Get the status of the workflow", Encoding.UTF8, "text/plain"); + using HttpResponseMessage statusResponse = await s_sharedHttpClient.PostAsync(statusUri, content); + Assert.True( + statusResponse.IsSuccessStatusCode, + $"Status check failed with status: {statusResponse.StatusCode}"); + string statusText = await statusResponse.Content.ReadAsStringAsync(); + this._outputHelper.WriteLine($"Status text: {statusText}"); + + bool isCompleted = statusText.Contains("Completed", StringComparison.OrdinalIgnoreCase); + bool hasContent = statusText.Contains( + "The Future of Artificial Intelligence", + StringComparison.OrdinalIgnoreCase); + return isCompleted && hasContent; + }, + message: "Orchestration is completed", + timeout: TimeSpan.FromSeconds(60)); + }); + } + + [Fact] + public async Task AgentAsMcpToolAsync() + { + string samplePath = Path.Combine(s_samplesPath, "07_AgentAsMcpTool"); + await this.RunSampleTestAsync(samplePath, async (logs) => + { + IClientTransport clientTransport = new HttpClientTransport(new() + { + Endpoint = new Uri($"http://localhost:{AzureFunctionsPort}/runtime/webhooks/mcp") + }); + + await using McpClient mcpClient = await McpClient.CreateAsync(clientTransport!); + + // Ensure the expected tools are present. + IList tools = await mcpClient.ListToolsAsync(); + + Assert.Single(tools, t => t.Name == "StockAdvisor"); + Assert.Single(tools, t => t.Name == "PlantAdvisor"); + + // Invoke the tools to verify they work as expected. + string stockPriceResponse = await this.InvokeMcpToolAsync(mcpClient, "StockAdvisor", "MSFT ATH"); + string plantSuggestionResponse = await this.InvokeMcpToolAsync(mcpClient, "PlantAdvisor", "Low light plant"); + Assert.NotEmpty(stockPriceResponse); + Assert.NotEmpty(plantSuggestionResponse); + + // Wait for up to 30 seconds to see if the agent responses are available in the logs + await this.WaitForConditionAsync( + condition: () => + { + lock (logs) + { + bool expectedLogsPresent = logs.Count(log => log.Message.Contains("Response:")) >= 2; + return Task.FromResult(expectedLogsPresent); + } + }, + message: "Agent response is available", + timeout: TimeSpan.FromSeconds(30)); + }); + } + + private async Task InvokeMcpToolAsync(McpClient mcpClient, string toolName, string query) + { + this._outputHelper.WriteLine($"Invoking MCP tool '{toolName}'..."); + + CallToolResult result = await mcpClient.CallToolAsync( + toolName, + arguments: new Dictionary { { "query", query } }); + + string toolCallResult = ((TextContentBlock)result.Content[0]).Text; + this._outputHelper.WriteLine($"MCP tool '{toolName}' response: {toolCallResult}"); + + return toolCallResult; + } + + private async Task TestSpamDetectionAsync(string emailId, string emailContent, bool expectedSpam) + { + object requestBody = new + { + email_id = emailId, + email_content = emailContent + }; + + string jsonContent = JsonSerializer.Serialize(requestBody); + using HttpContent content = new StringContent(jsonContent, Encoding.UTF8, "application/json"); + + Uri startUri = new($"http://localhost:{AzureFunctionsPort}/api/spamdetection/run"); + this._outputHelper.WriteLine($"Starting spam detection orchestration via POST request to {startUri}..."); + using HttpResponseMessage startResponse = await s_sharedHttpClient.PostAsync(startUri, content); + + Assert.True(startResponse.IsSuccessStatusCode, $"Start orchestration failed with status: {startResponse.StatusCode}"); + string startResponseText = await startResponse.Content.ReadAsStringAsync(); + JsonElement startResult = JsonElement.Parse(startResponseText); + + Assert.True(startResult.TryGetProperty("statusQueryGetUri", out JsonElement statusUriElement)); + Uri statusUri = new(statusUriElement.GetString()!); + + // Wait for orchestration to complete + await this.WaitForOrchestrationCompletionAsync(statusUri); + + // Verify the final result + using HttpResponseMessage statusResponse = await s_sharedHttpClient.GetAsync(statusUri); + Assert.True(statusResponse.IsSuccessStatusCode, $"Status check failed with status: {statusResponse.StatusCode}"); + + string statusText = await statusResponse.Content.ReadAsStringAsync(); + JsonElement statusResult = JsonElement.Parse(statusText); + + Assert.Equal("Completed", statusResult.GetProperty("runtimeStatus").GetString()); + Assert.True(statusResult.TryGetProperty("output", out JsonElement outputElement)); + + string output = outputElement.GetString()!; + Assert.NotEmpty(output); + + if (expectedSpam) + { + Assert.Contains("spam", output, StringComparison.OrdinalIgnoreCase); + } + else + { + Assert.Contains("sent", output, StringComparison.OrdinalIgnoreCase); + } + } + + private async Task StartSharedInfrastructureAsync() + { + // Start Azurite if it's not already running + if (!await this.IsAzuriteRunningAsync()) + { + await this.StartDockerContainerAsync( + containerName: "azurite", + image: "mcr.microsoft.com/azure-storage/azurite", + ports: ["-p", "10000:10000", "-p", "10001:10001", "-p", "10002:10002"]); + + // Wait for Azurite + await this.WaitForConditionAsync(this.IsAzuriteRunningAsync, "Azurite is running", TimeSpan.FromSeconds(30)); + } + + // Start DTS emulator if it's not already running + if (!await this.IsDtsEmulatorRunningAsync()) + { + await this.StartDockerContainerAsync( + containerName: "dts-emulator", + image: "mcr.microsoft.com/dts/dts-emulator:latest", + ports: ["-p", "8080:8080", "-p", "8082:8082"]); + + // Wait for DTS emulator + await this.WaitForConditionAsync( + condition: this.IsDtsEmulatorRunningAsync, + message: "DTS emulator is running", + timeout: TimeSpan.FromSeconds(30)); + } + } + + private async Task IsAzuriteRunningAsync() + { + this._outputHelper.WriteLine( + $"Checking if Azurite is running at http://localhost:{AzuritePort}/devstoreaccount1..."); + + try + { + using CancellationTokenSource timeoutCts = new(TimeSpan.FromSeconds(30)); + + // Example output when pinging Azurite: + // $ curl -i http://localhost:10000/devstoreaccount1?comp=list + // HTTP/1.1 403 Server failed to authenticate the request. + // Server: Azurite-Blob/3.34.0 + // x-ms-error-code: AuthorizationFailure + // x-ms-request-id: 6cd21522-bb0f-40f6-962c-fa174f17aa30 + // content-type: application/xml + // Date: Mon, 20 Oct 2025 23:52:02 GMT + // Connection: keep-alive + // Keep-Alive: timeout=5 + // Transfer-Encoding: chunked + using HttpResponseMessage response = await s_sharedHttpClient.GetAsync( + requestUri: new Uri($"http://localhost:{AzuritePort}/devstoreaccount1?comp=list"), + cancellationToken: timeoutCts.Token); + if (response.Headers.TryGetValues( + "Server", + out IEnumerable? serverValues) && serverValues.Any(s => s.StartsWith("Azurite", StringComparison.OrdinalIgnoreCase))) + { + this._outputHelper.WriteLine($"Azurite is running, server: {string.Join(", ", serverValues)}"); + return true; + } + + this._outputHelper.WriteLine($"Azurite is not running. Status code: {response.StatusCode}"); + return false; + } + catch (HttpRequestException ex) + { + this._outputHelper.WriteLine($"Azurite is not running: {ex.Message}"); + return false; + } + } + + private async Task IsDtsEmulatorRunningAsync() + { + this._outputHelper.WriteLine($"Checking if DTS emulator is running at http://localhost:{DtsPort}/healthz..."); + + // DTS emulator doesn't support HTTP/1.1, so we need to use HTTP/2.0 + using HttpClient http2Client = new() + { + DefaultRequestVersion = new Version(2, 0), + DefaultVersionPolicy = HttpVersionPolicy.RequestVersionExact + }; + + try + { + using CancellationTokenSource timeoutCts = new(TimeSpan.FromSeconds(30)); + using HttpResponseMessage response = await http2Client.GetAsync(new Uri($"http://localhost:{DtsPort}/healthz"), timeoutCts.Token); + if (response.Content.Headers.ContentLength > 0) + { + string content = await response.Content.ReadAsStringAsync(timeoutCts.Token); + this._outputHelper.WriteLine($"DTS emulator health check response: {content}"); + } + + if (response.IsSuccessStatusCode) + { + this._outputHelper.WriteLine("DTS emulator is running"); + return true; + } + + this._outputHelper.WriteLine($"DTS emulator is not running. Status code: {response.StatusCode}"); + return false; + } + catch (HttpRequestException ex) + { + this._outputHelper.WriteLine($"DTS emulator is not running: {ex.Message}"); + return false; + } + } + + private async Task StartDockerContainerAsync(string containerName, string image, string[] ports) + { + // Stop existing container if it exists + await this.RunCommandAsync("docker", ["stop", containerName]); + await this.RunCommandAsync("docker", ["rm", containerName]); + + // Start new container + List args = ["run", "-d", "--name", containerName]; + args.AddRange(ports); + args.Add(image); + + this._outputHelper.WriteLine( + $"Starting new container: {containerName} with image: {image} and ports: {string.Join(", ", ports)}"); + await this.RunCommandAsync("docker", args.ToArray()); + this._outputHelper.WriteLine($"Container started: {containerName}"); + } + + private async Task WaitForConditionAsync(Func> condition, string message, TimeSpan timeout) + { + this._outputHelper.WriteLine($"Waiting for '{message}'..."); + + using CancellationTokenSource cancellationTokenSource = new(timeout); + while (true) + { + if (await condition()) + { + return; + } + + try + { + await Task.Delay(TimeSpan.FromSeconds(1), cancellationTokenSource.Token); + } + catch (OperationCanceledException) when (cancellationTokenSource.IsCancellationRequested) + { + throw new TimeoutException($"Timeout waiting for '{message}'"); + } + } + } + + private async Task RunSampleTestAsync(string samplePath, Func, Task> testAction) + { + // Start the Azure Functions app + List logsContainer = []; + using Process funcProcess = this.StartFunctionApp(samplePath, logsContainer); + try + { + // Wait for the app to be ready + await this.WaitForAzureFunctionsAsync(); + + // Run the test + await testAction(logsContainer); + } + finally + { + await this.StopProcessAsync(funcProcess); + } + } + + private sealed record OutputLog(DateTime Timestamp, LogLevel Level, string Message); + + private Process StartFunctionApp(string samplePath, List logs) + { + ProcessStartInfo startInfo = new() + { + FileName = "dotnet", + Arguments = $"run -f {s_dotnetTargetFramework} --port {AzureFunctionsPort}", + WorkingDirectory = samplePath, + UseShellExecute = false, + RedirectStandardOutput = true, + RedirectStandardError = true, + }; + + string openAiEndpoint = s_configuration["AZURE_OPENAI_ENDPOINT"] ?? + throw new InvalidOperationException("The required AZURE_OPENAI_ENDPOINT env variable is not set."); + string openAiDeployment = s_configuration["AZURE_OPENAI_CHAT_DEPLOYMENT_NAME"] ?? + throw new InvalidOperationException("The required AZURE_OPENAI_CHAT_DEPLOYMENT_NAME env variable is not set."); + + // Set required environment variables for the function app (see local.settings.json for required settings) + startInfo.EnvironmentVariables["AZURE_OPENAI_ENDPOINT"] = openAiEndpoint; + startInfo.EnvironmentVariables["AZURE_OPENAI_DEPLOYMENT"] = openAiDeployment; + startInfo.EnvironmentVariables["DURABLE_TASK_SCHEDULER_CONNECTION_STRING"] = + $"Endpoint=http://localhost:{DtsPort};TaskHub=default;Authentication=None"; + startInfo.EnvironmentVariables["AzureWebJobsStorage"] = "UseDevelopmentStorage=true"; + + Process process = new() { StartInfo = startInfo }; + + // Capture the output and error streams + process.ErrorDataReceived += (sender, e) => + { + if (e.Data != null) + { + this._outputHelper.WriteLine($"[{startInfo.FileName}(err)]: {e.Data}"); + lock (logs) + { + logs.Add(new OutputLog(DateTime.Now, LogLevel.Error, e.Data)); + } + } + }; + + process.OutputDataReceived += (sender, e) => + { + if (e.Data != null) + { + this._outputHelper.WriteLine($"[{startInfo.FileName}(out)]: {e.Data}"); + lock (logs) + { + logs.Add(new OutputLog(DateTime.Now, LogLevel.Information, e.Data)); + } + } + }; + + if (!process.Start()) + { + throw new InvalidOperationException("Failed to start the function app"); + } + + process.BeginErrorReadLine(); + process.BeginOutputReadLine(); + + return process; + } + + private async Task WaitForAzureFunctionsAsync() + { + this._outputHelper.WriteLine( + $"Waiting for Azure Functions Core Tools to be ready at http://localhost:{AzureFunctionsPort}/..."); + await this.WaitForConditionAsync( + condition: async () => + { + try + { + using HttpRequestMessage request = new(HttpMethod.Head, $"http://localhost:{AzureFunctionsPort}/"); + using HttpResponseMessage response = await s_sharedHttpClient.SendAsync(request); + this._outputHelper.WriteLine($"Azure Functions Core Tools response: {response.StatusCode}"); + return response.IsSuccessStatusCode; + } + catch (HttpRequestException) + { + // Expected when the app isn't yet ready + return false; + } + }, + message: "Azure Functions Core Tools is ready", + timeout: TimeSpan.FromSeconds(60)); + } + + private async Task WaitForOrchestrationCompletionAsync(Uri statusUri) + { + using CancellationTokenSource timeoutCts = new(s_orchestrationTimeout); + while (true) + { + try + { + using HttpResponseMessage response = await s_sharedHttpClient.GetAsync( + statusUri, + timeoutCts.Token); + if (response.IsSuccessStatusCode) + { + string responseText = await response.Content.ReadAsStringAsync(timeoutCts.Token); + JsonElement result = JsonElement.Parse(responseText); + + if (result.TryGetProperty("runtimeStatus", out JsonElement statusElement) && + statusElement.GetString() is "Completed" or "Failed" or "Terminated") + { + return; + } + } + } + catch (Exception ex) when (!timeoutCts.Token.IsCancellationRequested) + { + // Ignore errors and retry + this._outputHelper.WriteLine($"Error waiting for orchestration completion: {ex}"); + } + + await Task.Delay(TimeSpan.FromSeconds(1), timeoutCts.Token); + } + } + + private async Task RunCommandAsync(string command, string[] args) + { + await this.RunCommandAsync(command, workingDirectory: null, args: args); + } + + private async Task RunCommandAsync(string command, string? workingDirectory, string[] args) + { + ProcessStartInfo startInfo = new() + { + FileName = command, + Arguments = string.Join(" ", args), + WorkingDirectory = workingDirectory, + UseShellExecute = false, + RedirectStandardOutput = true, + RedirectStandardError = true, + CreateNoWindow = true + }; + + this._outputHelper.WriteLine($"Running command: {command} {string.Join(" ", args)}"); + + using Process process = new() { StartInfo = startInfo }; + process.ErrorDataReceived += (sender, e) => this._outputHelper.WriteLine($"[{command}(err)]: {e.Data}"); + process.OutputDataReceived += (sender, e) => this._outputHelper.WriteLine($"[{command}(out)]: {e.Data}"); + if (!process.Start()) + { + throw new InvalidOperationException("Failed to start the command"); + } + process.BeginErrorReadLine(); + process.BeginOutputReadLine(); + + using CancellationTokenSource cancellationTokenSource = new(TimeSpan.FromMinutes(1)); + await process.WaitForExitAsync(cancellationTokenSource.Token); + + this._outputHelper.WriteLine($"Command completed with exit code: {process.ExitCode}"); + } + + private async Task StopProcessAsync(Process process) + { + try + { + if (!process.HasExited) + { + this._outputHelper.WriteLine($"Killing process {process.ProcessName}#{process.Id}"); + process.Kill(entireProcessTree: true); + + using CancellationTokenSource timeoutCts = new(TimeSpan.FromSeconds(10)); + await process.WaitForExitAsync(timeoutCts.Token); + this._outputHelper.WriteLine($"Process exited: {process.Id}"); + } + } + catch (Exception ex) + { + this._outputHelper.WriteLine($"Failed to stop process: {ex.Message}"); + } + } + + private static string GetTargetFramework() + { + // Get the target framework by looking at the path of the current file. It should be something like /path/to/project/bin/Debug/net8.0/... + string filePath = new Uri(typeof(SamplesValidation).Assembly.Location).LocalPath; + string directory = Path.GetDirectoryName(filePath)!; + string tfm = Path.GetFileName(directory); + if (tfm.StartsWith("net", StringComparison.OrdinalIgnoreCase)) + { + return tfm; + } + + throw new InvalidOperationException($"Unable to find target framework in path: {filePath}"); + } +} diff --git a/dotnet/tests/Microsoft.Agents.AI.Hosting.AzureFunctions.UnitTests/DurableAgentFunctionMetadataTransformerTests.cs b/dotnet/tests/Microsoft.Agents.AI.Hosting.AzureFunctions.UnitTests/DurableAgentFunctionMetadataTransformerTests.cs new file mode 100644 index 0000000000..7d3a2ec13e --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.Hosting.AzureFunctions.UnitTests/DurableAgentFunctionMetadataTransformerTests.cs @@ -0,0 +1,186 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Diagnostics.CodeAnalysis; +using Microsoft.Azure.Functions.Worker.Core.FunctionMetadata; +using Microsoft.Extensions.Logging.Abstractions; + +namespace Microsoft.Agents.AI.Hosting.AzureFunctions.UnitTests; + +public sealed class DurableAgentFunctionMetadataTransformerTests +{ + [Theory] + [InlineData(0, false, false, 1)] // entity only + [InlineData(0, true, false, 2)] // entity + http + [InlineData(0, false, true, 2)] // entity + mcp tool + [InlineData(0, true, true, 3)] // entity + http + mcp tool + [InlineData(3, true, true, 3)] // entity + http + mcp tool added to existing + public void Transform_AddsAgentAndHttpTriggers_ForEachAgent( + int initialMetadataEntryCount, + bool enableHttp, + bool enableMcp, + int expectedMetadataCount) + { + // Arrange + Dictionary> agents = new() + { + { "testAgent", _ => new TestAgent("testAgent", "Test agent description") } + }; + + FunctionsAgentOptions options = new(); + + options.HttpTrigger.IsEnabled = enableHttp; + options.McpToolTrigger.IsEnabled = enableMcp; + + IFunctionsAgentOptionsProvider agentOptionsProvider = new FakeOptionsProvider(new Dictionary + { + { "testAgent", options } + }); + + List metadataList = BuildFunctionMetadataList(initialMetadataEntryCount); + + DurableAgentFunctionMetadataTransformer transformer = new( + agents, + NullLogger.Instance, + new FakeServiceProvider(), + agentOptionsProvider); + + // Act + transformer.Transform(metadataList); + + // Assert + Assert.Equal(initialMetadataEntryCount + expectedMetadataCount, metadataList.Count); + + DefaultFunctionMetadata agentTrigger = Assert.IsType(metadataList[initialMetadataEntryCount]); + Assert.Equal("dafx-testAgent", agentTrigger.Name); + Assert.Contains("entityTrigger", agentTrigger.RawBindings![0]); + + if (enableHttp) + { + DefaultFunctionMetadata httpTrigger = Assert.IsType(metadataList[initialMetadataEntryCount + 1]); + Assert.Equal("http-testAgent", httpTrigger.Name); + Assert.Contains("httpTrigger", httpTrigger.RawBindings![0]); + } + + if (enableMcp) + { + int mcpIndex = initialMetadataEntryCount + (enableHttp ? 2 : 1); + DefaultFunctionMetadata mcpToolTrigger = Assert.IsType(metadataList[mcpIndex]); + Assert.Equal("mcptool-testAgent", mcpToolTrigger.Name); + Assert.Contains("mcpToolTrigger", mcpToolTrigger.RawBindings![0]); + } + } + + [Fact] + public void Transform_AddsTriggers_ForMultipleAgents() + { + // Arrange + Dictionary> agents = new() + { + { "agentA", _ => new TestAgent("testAgentA", "Test agent description") }, + { "agentB", _ => new TestAgent("testAgentB", "Test agent description") }, + { "agentC", _ => new TestAgent("testAgentC", "Test agent description") } + }; + + // Helper to create options with configurable triggers + static FunctionsAgentOptions CreateFunctionsAgentOptions(bool httpEnabled, bool mcpEnabled) + { + FunctionsAgentOptions options = new(); + options.HttpTrigger.IsEnabled = httpEnabled; + options.McpToolTrigger.IsEnabled = mcpEnabled; + return options; + } + + FunctionsAgentOptions agentOptionsA = CreateFunctionsAgentOptions(true, false); + FunctionsAgentOptions agentOptionsB = CreateFunctionsAgentOptions(true, true); + FunctionsAgentOptions agentOptionsC = CreateFunctionsAgentOptions(true, true); + + Dictionary functionsAgentOptions = new() + { + { "agentA", agentOptionsA }, + { "agentB", agentOptionsB }, + { "agentC", agentOptionsC } + }; + + IFunctionsAgentOptionsProvider agentOptionsProvider = new FakeOptionsProvider(functionsAgentOptions); + DurableAgentFunctionMetadataTransformer transformer = new( + agents, + NullLogger.Instance, + new FakeServiceProvider(), + agentOptionsProvider); + + const int InitialMetadataEntryCount = 2; + List metadataList = BuildFunctionMetadataList(InitialMetadataEntryCount); + + // Act + transformer.Transform(metadataList); + + // Assert + Assert.Equal(InitialMetadataEntryCount + (agents.Count * 2) + 2, metadataList.Count); + + foreach (string agentName in agents.Keys) + { + // The agent's entity trigger name is prefixed with "dafx-" + DefaultFunctionMetadata entityMeta = + Assert.IsType( + Assert.Single(metadataList, m => m.Name == $"dafx-{agentName}")); + Assert.NotNull(entityMeta.RawBindings); + Assert.Contains("entityTrigger", entityMeta.RawBindings[0]); + + DefaultFunctionMetadata httpMeta = + Assert.IsType( + Assert.Single(metadataList, m => m.Name == $"http-{agentName}")); + Assert.NotNull(httpMeta.RawBindings); + Assert.Contains("httpTrigger", httpMeta.RawBindings[0]); + Assert.Contains($"agents/{agentName}/run", httpMeta.RawBindings[0]); + + // We expect 2 mcp tool triggers only for agentB and agentC + if (agentName is "agentB" or "agentC") + { + DefaultFunctionMetadata? mcpToolMeta = + Assert.Single(metadataList, m => m.Name == $"mcptool-{agentName}") as DefaultFunctionMetadata; + Assert.NotNull(mcpToolMeta); + Assert.NotNull(mcpToolMeta.RawBindings); + Assert.Equal(4, mcpToolMeta.RawBindings.Count); + Assert.Contains("mcpToolTrigger", mcpToolMeta.RawBindings[0]); + Assert.Contains("mcpToolProperty", mcpToolMeta.RawBindings[1]); // We expect 2 tool property bindings + Assert.Contains("mcpToolProperty", mcpToolMeta.RawBindings[2]); + } + } + } + + private static List BuildFunctionMetadataList(int numberOfFunctions) + { + List list = []; + for (int i = 0; i < numberOfFunctions; i++) + { + list.Add(new DefaultFunctionMetadata + { + Language = "dotnet-isolated", + Name = $"SingleAgentOrchestration{i + 1}", + EntryPoint = "MyApp.Functions.SingleAgentOrchestration", + RawBindings = ["{\r\n \"name\": \"context\",\r\n \"direction\": \"In\",\r\n \"type\": \"orchestrationTrigger\",\r\n \"properties\": {}\r\n }"], + ScriptFile = "MyApp.dll" + }); + } + + return list; + } + + private sealed class FakeServiceProvider : IServiceProvider + { + public object? GetService(Type serviceType) => null; + } + + private sealed class FakeOptionsProvider : IFunctionsAgentOptionsProvider + { + private readonly Dictionary _map; + + public FakeOptionsProvider(Dictionary map) + { + this._map = map ?? throw new ArgumentNullException(nameof(map)); + } + + public bool TryGet(string agentName, [NotNullWhen(true)] out FunctionsAgentOptions? options) + => this._map.TryGetValue(agentName, out options); + } +} diff --git a/dotnet/tests/Microsoft.Agents.AI.Hosting.AzureFunctions.UnitTests/Microsoft.Agents.AI.Hosting.AzureFunctions.UnitTests.csproj b/dotnet/tests/Microsoft.Agents.AI.Hosting.AzureFunctions.UnitTests/Microsoft.Agents.AI.Hosting.AzureFunctions.UnitTests.csproj new file mode 100644 index 0000000000..7b053abe83 --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.Hosting.AzureFunctions.UnitTests/Microsoft.Agents.AI.Hosting.AzureFunctions.UnitTests.csproj @@ -0,0 +1,12 @@ + + + + $(TargetFrameworksCore) + enable + + + + + + + diff --git a/dotnet/tests/Microsoft.Agents.AI.Hosting.AzureFunctions.UnitTests/TestAgent.cs b/dotnet/tests/Microsoft.Agents.AI.Hosting.AzureFunctions.UnitTests/TestAgent.cs new file mode 100644 index 0000000000..b0ad7ec0fe --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.Hosting.AzureFunctions.UnitTests/TestAgent.cs @@ -0,0 +1,33 @@ +// Copyright (c) Microsoft. All rights reserved. + +using System.Text.Json; +using Microsoft.Extensions.AI; + +namespace Microsoft.Agents.AI.Hosting.AzureFunctions.UnitTests; + +internal sealed class TestAgent(string name, string description) : AIAgent +{ + public override string? Name => name; + + public override string? Description => description; + + public override AgentThread GetNewThread() => new DummyAgentThread(); + + public override AgentThread DeserializeThread( + JsonElement serializedThread, + JsonSerializerOptions? jsonSerializerOptions = null) => new DummyAgentThread(); + + public override Task RunAsync( + IEnumerable messages, + AgentThread? thread = null, + AgentRunOptions? options = null, + CancellationToken cancellationToken = default) => Task.FromResult(new AgentRunResponse([.. messages])); + + public override IAsyncEnumerable RunStreamingAsync( + IEnumerable messages, + AgentThread? thread = null, + AgentRunOptions? options = null, + CancellationToken cancellationToken = default) => throw new NotSupportedException(); + + private sealed class DummyAgentThread : AgentThread; +} diff --git a/dotnet/tests/Microsoft.Agents.AI.Hosting.OpenAI.UnitTests/ConformanceTraces/ChatCompletions/tools/request.json b/dotnet/tests/Microsoft.Agents.AI.Hosting.OpenAI.UnitTests/ConformanceTraces/ChatCompletions/tools/request.json new file mode 100644 index 0000000000..b41ac7ab2e --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.Hosting.OpenAI.UnitTests/ConformanceTraces/ChatCompletions/tools/request.json @@ -0,0 +1,53 @@ +{ + "model": "gpt-4o-mini", + "messages": [ + { + "role": "user", + "content": "What's the weather like in San Francisco?" + } + ], + "max_completion_tokens": 256, + "temperature": 0.7, + "top_p": 1, + "tools": [ + { + "type": "function", + "function": { + "name": "get_weather", + "description": "Get the current weather in a given location", + "parameters": { + "type": "object", + "properties": { + "location": { + "type": "string", + "description": "The city and state, e.g. San Francisco, CA" + }, + "unit": { + "type": "string", + "enum": [ "celsius", "fahrenheit" ], + "description": "Temperature unit" + } + }, + "required": [ "location" ] + } + } + }, + { + "type": "function", + "function": { + "name": "get_time", + "description": "Get the current time in a given timezone", + "parameters": { + "type": "object", + "properties": { + "timezone": { + "type": "string", + "description": "The IANA timezone, e.g. America/Los_Angeles" + } + }, + "required": [ "timezone" ] + } + } + } + ] +} \ No newline at end of file diff --git a/dotnet/tests/Microsoft.Agents.AI.Hosting.OpenAI.UnitTests/ConformanceTraces/ChatCompletions/tools/response.json b/dotnet/tests/Microsoft.Agents.AI.Hosting.OpenAI.UnitTests/ConformanceTraces/ChatCompletions/tools/response.json new file mode 100644 index 0000000000..b86280bca0 --- /dev/null +++ b/dotnet/tests/Microsoft.Agents.AI.Hosting.OpenAI.UnitTests/ConformanceTraces/ChatCompletions/tools/response.json @@ -0,0 +1,42 @@ +{ + "id": "chatcmpl-tools-test-001", + "object": "chat.completion", + "created": 1234567890, + "model": "gpt-4o-mini", + "choices": [ + { + "index": 0, + "message": { + "role": "assistant", + "content": null, + "tool_calls": [ + { + "id": "call_abc123", + "type": "function", + "function": { + "name": "get_weather", + "arguments": "{\"location\": \"San Francisco, CA\", \"unit\": \"fahrenheit\"}" + } + } + ] + }, + "finish_reason": "tool_calls" + } + ], + "usage": { + "prompt_tokens": 85, + "completion_tokens": 32, + "total_tokens": 117, + "prompt_tokens_details": { + "cached_tokens": 0, + "audio_tokens": 0 + }, + "completion_tokens_details": { + "reasoning_tokens": 0, + "audio_tokens": 0, + "accepted_prediction_tokens": 0, + "rejected_prediction_tokens": 0 + } + }, + "service_tier": "default" +} \ No newline at end of file diff --git a/dotnet/tests/Microsoft.Agents.AI.Hosting.OpenAI.UnitTests/ContentTypeEventGeneratorTests.cs b/dotnet/tests/Microsoft.Agents.AI.Hosting.OpenAI.UnitTests/ContentTypeEventGeneratorTests.cs index 5a8f4ea442..1be9d06ca7 100644 --- a/dotnet/tests/Microsoft.Agents.AI.Hosting.OpenAI.UnitTests/ContentTypeEventGeneratorTests.cs +++ b/dotnet/tests/Microsoft.Agents.AI.Hosting.OpenAI.UnitTests/ContentTypeEventGeneratorTests.cs @@ -47,7 +47,7 @@ public async Task TextReasoningContent_GeneratesReasoningItem_SuccessAsync() var firstItemAddedEvent = events.First(e => e.GetProperty("type").GetString() == "response.output_item.added"); var firstItem = firstItemAddedEvent.GetProperty("item"); Assert.Equal("reasoning", firstItem.GetProperty("type").GetString()); - Assert.True(firstItemAddedEvent.GetProperty("output_index").GetInt32() == 0); + Assert.Equal(0, firstItemAddedEvent.GetProperty("output_index").GetInt32()); // Verify reasoning item done var firstItemDoneEvent = events.First(e => @@ -153,7 +153,7 @@ public async Task ErrorContent_GeneratesRefusalItem_SuccessAsync() // Verify item added event var itemAddedEvent = events.FirstOrDefault(e => e.GetProperty("type").GetString() == "response.output_item.added"); - Assert.True(itemAddedEvent.ValueKind != JsonValueKind.Undefined); + Assert.NotEqual(JsonValueKind.Undefined, itemAddedEvent.ValueKind); var item = itemAddedEvent.GetProperty("item"); Assert.Equal("message", item.GetProperty("type").GetString()); @@ -166,7 +166,7 @@ public async Task ErrorContent_GeneratesRefusalItem_SuccessAsync() Assert.NotEmpty(contentArray); var refusalContent = contentArray.First(c => c.GetProperty("type").GetString() == "refusal"); - Assert.True(refusalContent.ValueKind != JsonValueKind.Undefined); + Assert.NotEqual(JsonValueKind.Undefined, refusalContent.ValueKind); Assert.Equal(ErrorMessage, refusalContent.GetProperty("refusal").GetString()); } @@ -246,12 +246,12 @@ public async Task ImageContent_UriContent_GeneratesImageItem_SuccessAsync() // Assert var itemAddedEvent = events.FirstOrDefault(e => e.GetProperty("type").GetString() == "response.output_item.added"); - Assert.True(itemAddedEvent.ValueKind != JsonValueKind.Undefined); + Assert.NotEqual(JsonValueKind.Undefined, itemAddedEvent.ValueKind); var content = itemAddedEvent.GetProperty("item").GetProperty("content"); var imageContent = content.EnumerateArray().First(c => c.GetProperty("type").GetString() == "input_image"); - Assert.True(imageContent.ValueKind != JsonValueKind.Undefined); + Assert.NotEqual(JsonValueKind.Undefined, imageContent.ValueKind); Assert.Equal(ImageUrl, imageContent.GetProperty("image_url").GetString()); } @@ -270,12 +270,12 @@ public async Task ImageContent_DataContent_GeneratesImageItem_SuccessAsync() // Assert var itemAddedEvent = events.FirstOrDefault(e => e.GetProperty("type").GetString() == "response.output_item.added"); - Assert.True(itemAddedEvent.ValueKind != JsonValueKind.Undefined); + Assert.NotEqual(JsonValueKind.Undefined, itemAddedEvent.ValueKind); var content = itemAddedEvent.GetProperty("item").GetProperty("content"); var imageContent = content.EnumerateArray().First(c => c.GetProperty("type").GetString() == "input_image"); - Assert.True(imageContent.ValueKind != JsonValueKind.Undefined); + Assert.NotEqual(JsonValueKind.Undefined, imageContent.ValueKind); Assert.Equal(DataUri, imageContent.GetProperty("image_url").GetString()); } @@ -295,12 +295,12 @@ public async Task ImageContent_WithDetailProperty_IncludesDetail_SuccessAsync() // Assert var itemAddedEvent = events.FirstOrDefault(e => e.GetProperty("type").GetString() == "response.output_item.added"); - Assert.True(itemAddedEvent.ValueKind != JsonValueKind.Undefined); + Assert.NotEqual(JsonValueKind.Undefined, itemAddedEvent.ValueKind); var content = itemAddedEvent.GetProperty("item").GetProperty("content"); var imageContent = content.EnumerateArray().First(c => c.GetProperty("type").GetString() == "input_image"); - Assert.True(imageContent.ValueKind != JsonValueKind.Undefined); + Assert.NotEqual(JsonValueKind.Undefined, imageContent.ValueKind); Assert.True(imageContent.TryGetProperty("detail", out var detailProp)); Assert.Equal(Detail, detailProp.GetString()); } @@ -345,12 +345,12 @@ public async Task AudioContent_Mp3Format_GeneratesAudioItem_SuccessAsync() // Assert var itemAddedEvent = events.FirstOrDefault(e => e.GetProperty("type").GetString() == "response.output_item.added"); - Assert.True(itemAddedEvent.ValueKind != JsonValueKind.Undefined); + Assert.NotEqual(JsonValueKind.Undefined, itemAddedEvent.ValueKind); var content = itemAddedEvent.GetProperty("item").GetProperty("content"); var audioContent = content.EnumerateArray().First(c => c.GetProperty("type").GetString() == "input_audio"); - Assert.True(audioContent.ValueKind != JsonValueKind.Undefined); + Assert.NotEqual(JsonValueKind.Undefined, audioContent.ValueKind); Assert.Equal(AudioDataUri, audioContent.GetProperty("data").GetString()); Assert.Equal("mp3", audioContent.GetProperty("format").GetString()); } @@ -421,12 +421,12 @@ public async Task HostedFileContent_GeneratesFileItem_SuccessAsync() // Assert var itemAddedEvent = events.FirstOrDefault(e => e.GetProperty("type").GetString() == "response.output_item.added"); - Assert.True(itemAddedEvent.ValueKind != JsonValueKind.Undefined); + Assert.NotEqual(JsonValueKind.Undefined, itemAddedEvent.ValueKind); var content = itemAddedEvent.GetProperty("item").GetProperty("content"); var fileContent = content.EnumerateArray().First(c => c.GetProperty("type").GetString() == "input_file"); - Assert.True(fileContent.ValueKind != JsonValueKind.Undefined); + Assert.NotEqual(JsonValueKind.Undefined, fileContent.ValueKind); Assert.Equal(FileId, fileContent.GetProperty("file_id").GetString()); } @@ -471,12 +471,12 @@ public async Task FileContent_WithDataUri_GeneratesFileItem_SuccessAsync() // Assert var itemAddedEvent = events.FirstOrDefault(e => e.GetProperty("type").GetString() == "response.output_item.added"); - Assert.True(itemAddedEvent.ValueKind != JsonValueKind.Undefined); + Assert.NotEqual(JsonValueKind.Undefined, itemAddedEvent.ValueKind); var content = itemAddedEvent.GetProperty("item").GetProperty("content"); var fileContent = content.EnumerateArray().First(c => c.GetProperty("type").GetString() == "input_file"); - Assert.True(fileContent.ValueKind != JsonValueKind.Undefined); + Assert.NotEqual(JsonValueKind.Undefined, fileContent.ValueKind); Assert.Equal(FileDataUri, fileContent.GetProperty("file_data").GetString()); Assert.Equal(Filename, fileContent.GetProperty("filename").GetString()); } @@ -499,7 +499,7 @@ public async Task FileContent_WithoutFilename_GeneratesFileItemWithoutFilename_S var content = itemAddedEvent.GetProperty("item").GetProperty("content"); var fileContent = content.EnumerateArray().First(c => c.GetProperty("type").GetString() == "input_file"); - Assert.True(fileContent.ValueKind != JsonValueKind.Undefined); + Assert.NotEqual(JsonValueKind.Undefined, fileContent.ValueKind); Assert.Equal(FileDataUri, fileContent.GetProperty("file_data").GetString()); // filename property might be null or absent } diff --git a/dotnet/tests/Microsoft.Agents.AI.Hosting.OpenAI.UnitTests/EndpointRouteBuilderExtensionsTests.cs b/dotnet/tests/Microsoft.Agents.AI.Hosting.OpenAI.UnitTests/EndpointRouteBuilderExtensionsTests.cs index 38b13fbfac..e3effac07a 100644 --- a/dotnet/tests/Microsoft.Agents.AI.Hosting.OpenAI.UnitTests/EndpointRouteBuilderExtensionsTests.cs +++ b/dotnet/tests/Microsoft.Agents.AI.Hosting.OpenAI.UnitTests/EndpointRouteBuilderExtensionsTests.cs @@ -223,4 +223,174 @@ public void MapOpenAIResponses_WithoutAgent_CustomPath_Succeeds() app.MapOpenAIResponses(responsesPath: "/custom/path/responses"); Assert.NotNull(app); } + + /// + /// Verifies that MapOpenAIResponses throws ArgumentNullException for null endpoints when using IHostedAgentBuilder. + /// + [Fact] + public void MapOpenAIResponses_WithAgentBuilder_NullEndpoints_ThrowsArgumentNullException() + { + // Arrange + AspNetCore.Routing.IEndpointRouteBuilder endpoints = null!; + WebApplicationBuilder builder = WebApplication.CreateBuilder(); + IChatClient mockChatClient = new TestHelpers.SimpleMockChatClient(); + builder.Services.AddKeyedSingleton("chat-client", mockChatClient); + IHostedAgentBuilder agentBuilder = builder.AddAIAgent("agent", "Instructions", chatClientServiceKey: "chat-client"); + + // Act & Assert + ArgumentNullException exception = Assert.Throws(() => + endpoints.MapOpenAIResponses(agentBuilder)); + + Assert.Equal("endpoints", exception.ParamName); + } + + /// + /// Verifies that MapOpenAIResponses throws ArgumentNullException for null agentBuilder. + /// + [Fact] + public void MapOpenAIResponses_WithAgentBuilder_NullAgentBuilder_ThrowsArgumentNullException() + { + // Arrange + WebApplicationBuilder builder = WebApplication.CreateBuilder(); + IChatClient mockChatClient = new TestHelpers.SimpleMockChatClient(); + builder.Services.AddKeyedSingleton("chat-client", mockChatClient); + builder.AddOpenAIResponses(); + using WebApplication app = builder.Build(); + IHostedAgentBuilder agentBuilder = null!; + + // Act & Assert + ArgumentNullException exception = Assert.Throws(() => + app.MapOpenAIResponses(agentBuilder)); + + Assert.Equal("agentBuilder", exception.ParamName); + } + + /// + /// Verifies that MapOpenAIResponses with IHostedAgentBuilder correctly resolves and maps the agent. + /// + [Fact] + public void MapOpenAIResponses_WithAgentBuilder_Succeeds() + { + // Arrange + WebApplicationBuilder builder = WebApplication.CreateBuilder(); + IChatClient mockChatClient = new TestHelpers.SimpleMockChatClient(); + builder.Services.AddKeyedSingleton("chat-client", mockChatClient); + IHostedAgentBuilder agentBuilder = builder.AddAIAgent("agent", "Instructions", chatClientServiceKey: "chat-client"); + builder.AddOpenAIResponses(); + using WebApplication app = builder.Build(); + + // Act & Assert - Should not throw + app.MapOpenAIResponses(agentBuilder); + Assert.NotNull(app); + } + + /// + /// Verifies that MapOpenAIResponses with IHostedAgentBuilder and custom path works correctly. + /// + [Fact] + public void MapOpenAIResponses_WithAgentBuilder_CustomPath_Succeeds() + { + // Arrange + WebApplicationBuilder builder = WebApplication.CreateBuilder(); + IChatClient mockChatClient = new TestHelpers.SimpleMockChatClient(); + builder.Services.AddKeyedSingleton("chat-client", mockChatClient); + IHostedAgentBuilder agentBuilder = builder.AddAIAgent("my-agent", "Instructions", chatClientServiceKey: "chat-client"); + builder.AddOpenAIResponses(); + using WebApplication app = builder.Build(); + + // Act & Assert - Should not throw + app.MapOpenAIResponses(agentBuilder, path: "/agents/my-agent/responses"); + Assert.NotNull(app); + } + + /// + /// Verifies that multiple agents can be mapped using IHostedAgentBuilder. + /// + [Fact] + public void MapOpenAIResponses_WithAgentBuilder_MultipleAgents_Succeeds() + { + // Arrange + WebApplicationBuilder builder = WebApplication.CreateBuilder(); + IChatClient mockChatClient = new TestHelpers.SimpleMockChatClient(); + builder.Services.AddKeyedSingleton("chat-client", mockChatClient); + IHostedAgentBuilder agent1Builder = builder.AddAIAgent("agent1", "Instructions1", chatClientServiceKey: "chat-client"); + IHostedAgentBuilder agent2Builder = builder.AddAIAgent("agent2", "Instructions2", chatClientServiceKey: "chat-client"); + builder.AddOpenAIResponses(); + using WebApplication app = builder.Build(); + + // Act & Assert - Should not throw + app.MapOpenAIResponses(agent1Builder); + app.MapOpenAIResponses(agent2Builder); + Assert.NotNull(app); + } + + /// + /// Verifies that IHostedAgentBuilder overload validates agent name characters. + /// + [Theory] + [InlineData("agent with spaces")] + [InlineData("agent