mirror of
https://github.com/microsoft/agent-framework.git
synced 2026-06-16 21:04:09 +08:00
88ea9d08c7
* Initial plan * Update code for Microsoft.Extensions.AI.Abstractions 10.4.0 breaking changes - Rename FunctionApprovalRequestContent → ToolApprovalRequestContent - Rename FunctionApprovalResponseContent → ToolApprovalResponseContent - Rename UserInputRequestContent → ToolApprovalRequestContent - Rename UserInputResponseContent → ToolApprovalResponseContent - Update .FunctionCall property → .ToolCall with FunctionCallContent casts where needed - Update .Id property → .RequestId on the renamed types - Rename FunctionApprovalRequestEventGenerator → ToolApprovalRequestEventGenerator - Rename FunctionApprovalResponseEventGenerator → ToolApprovalResponseEventGenerator Co-authored-by: Copilot <223556219+Copilot@users.noreply.github.com> * Update OpenAI 2.9.1, ME.AI 10.4.0, fix breaking API changes Co-authored-by: stephentoub <2642209+stephentoub@users.noreply.github.com> * Fix remaining ME.AI 10.4.0 breaking changes: MCP approval types, .Output→.Outputs Co-authored-by: stephentoub <2642209+stephentoub@users.noreply.github.com> * Use pattern matching with `when` for ToolApprovalRequestContent/FunctionCallContent Co-authored-by: stephentoub <2642209+stephentoub@users.noreply.github.com> * Update Azure.AI.OpenAI to 2.9.0-beta.1 Co-authored-by: stephentoub <2642209+stephentoub@users.noreply.github.com> * Fix remaining GetResponsesClient(model) build failures for Azure.AI.OpenAI 2.9.0-beta.1 Co-authored-by: stephentoub <2642209+stephentoub@users.noreply.github.com> * Address review feedback: remove redundant type checks in TestRequestAgent.cs and fix error message in AIAgentHostExecutor.cs Co-authored-by: stephentoub <2642209+stephentoub@users.noreply.github.com> * Update Azure.AI.Projects to 2.0.0-beta.2 with namespace migration - Azure.AI.Projects 2.0.0-beta.1 → 2.0.0-beta.2 - Azure.AI.Projects.OpenAI → Azure.AI.Extensions.OpenAI (transitive) - Agent types moved to Azure.AI.Projects.Agents namespace - AgentRecord.Versions.Latest → AgentRecord.GetLatestVersion() - OpenAPIFunctionDefinition → OpenApiFunctionDefinition - BingCustomSearchToolParameters → BingCustomSearchToolOptions - MemorySearchPreviewTool.UpdateDelay → UpdateDelayInSecs - Azure.Identity 1.17.1 → 1.19.0 - Microsoft.Identity.Client.Extensions.Msal 4.78.0 → 4.83.1 Co-authored-by: rogerbarreto <19890735+rogerbarreto@users.noreply.github.com> * Fix remaining type renames for Azure.AI.Projects 2.0.0-beta.2 - BrowserAutomationToolParameters → BrowserAutomationToolOptions - MemoryUpdateOptions.UpdateDelay stays as UpdateDelay (not renamed) - WaitForMemoriesUpdateAsync parameter order: pollingInterval before options - AIProjectAgentsOperations → AgentsClient Co-authored-by: rogerbarreto <19890735+rogerbarreto@users.noreply.github.com> * Fix format errors and OpenTelemetry test for ME.AI 10.4.0 - Remove unused 'using Azure.AI.Extensions.OpenAI' and fix import ordering in Agent_With_AzureAIProject/Program.cs - Update OpenTelemetryAgentTests: gen_ai.tool.definitions is now always emitted regardless of EnableSensitiveData per ME.AI 10.4.0 change (dotnet/extensions#7346). Tool definitions are not considered sensitive. Co-authored-by: Copilot <223556219+Copilot@users.noreply.github.com> * Fix GetRepoFolder() to work in git worktrees Use 'workflow-samples' directory as repo root marker instead of '.git', which fails in worktrees (.git is a file) and also matches too early when a '.github' folder exists in subdirectories. Co-authored-by: Copilot <223556219+Copilot@users.noreply.github.com> * Fix formatting: remove unused usings and fix import ordering dotnet format applied across 59 impacted projects. Primarily removes unnecessary 'using Azure.AI.Projects' where Azure.AI.Projects.Agents provides all needed types, and fixes import ordering per editorconfig. Co-authored-by: Copilot <223556219+Copilot@users.noreply.github.com> * Disable AzureAIAgentsPersistent integration tests (#4769) Azure.AI.Agents.Persistent 1.2.0-beta.9 references McpServerToolApprovalResponseContent which was removed in ME.AI 10.4.0 (renamed to ToolApprovalResponseContent), causing TypeLoadException at runtime. Mark all 6 test classes with IntegrationDisabled trait until Persistent ships a version targeting ME.AI 10.4.0+. Upstream fix: https://github.com/Azure/azure-sdk-for-net/pull/56929 Co-authored-by: Copilot <223556219+Copilot@users.noreply.github.com> * Add README with compatibility note for AzureAI.Persistent (#4769) Documents that Azure.AI.Agents.Persistent 1.2.0-beta.9 is only compatible with ME.AI ≤10.3.0 and OpenAI ≤2.8.0 due to type renames in ME.AI 10.4.0. Co-authored-by: Copilot <223556219+Copilot@users.noreply.github.com> * Fix file encoding: restore UTF-8 BOM on Persistent test files Co-authored-by: Copilot <223556219+Copilot@users.noreply.github.com> * Mark AzureAI.Persistent as IsPackable=false (#4769) Prevent shipping until Azure.AI.Agents.Persistent targets ME.AI 10.4.0+. Co-authored-by: Copilot <223556219+Copilot@users.noreply.github.com> * Moving IsPackable after import --------- Co-authored-by: copilot-swe-agent[bot] <198982749+Copilot@users.noreply.github.com> Co-authored-by: Copilot <223556219+Copilot@users.noreply.github.com> Co-authored-by: stephentoub <2642209+stephentoub@users.noreply.github.com> Co-authored-by: rogerbarreto <19890735+rogerbarreto@users.noreply.github.com>
165 lines
7.6 KiB
C#
165 lines
7.6 KiB
C#
// Copyright (c) Microsoft. All rights reserved.
|
|
|
|
// This sample demonstrates how to use GroupChatBuilder with tools that require human
|
|
// approval before execution. A group of specialized agents collaborate on a task, and
|
|
// sensitive tool calls trigger human-in-the-loop approval.
|
|
//
|
|
// This sample works as follows:
|
|
// 1. A GroupChatBuilder workflow is created with multiple specialized agents.
|
|
// 2. A custom manager determines which agent speaks next based on conversation state.
|
|
// 3. Agents collaborate on a software deployment task.
|
|
// 4. When the deployment agent tries to deploy to production, it triggers an approval request.
|
|
// 5. The sample simulates human approval and the workflow completes.
|
|
//
|
|
// Purpose:
|
|
// Show how tool call approvals integrate with multi-agent group chat workflows where
|
|
// different agents have different levels of tool access.
|
|
//
|
|
// Demonstrate:
|
|
// - Using custom GroupChatManager with agents that have approval-required tools.
|
|
// - Handling ToolApprovalRequestContent in group chat scenarios.
|
|
// - Multi-round group chat with tool approval interruption and resumption.
|
|
|
|
using System.ComponentModel;
|
|
using System.Text.Json;
|
|
using Azure.AI.OpenAI;
|
|
using Azure.Identity;
|
|
using Microsoft.Agents.AI;
|
|
using Microsoft.Agents.AI.Workflows;
|
|
using Microsoft.Extensions.AI;
|
|
|
|
namespace WorkflowGroupChatToolApprovalSample;
|
|
|
|
/// <summary>
|
|
/// This sample demonstrates how to use GroupChatBuilder with tools that require human
|
|
/// approval before execution.
|
|
/// </summary>
|
|
/// <remarks>
|
|
/// Pre-requisites:
|
|
/// - An Azure OpenAI chat completion deployment must be configured.
|
|
/// </remarks>
|
|
public static class Program
|
|
{
|
|
private static async Task Main()
|
|
{
|
|
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";
|
|
|
|
// WARNING: DefaultAzureCredential is convenient for development but requires careful consideration in production.
|
|
// In production, consider using a specific credential (e.g., ManagedIdentityCredential) to avoid
|
|
// latency issues, unintended credential probing, and potential security risks from fallback mechanisms.
|
|
// 1. Create AI client
|
|
IChatClient client = new AzureOpenAIClient(new Uri(endpoint), new DefaultAzureCredential())
|
|
.GetChatClient(deploymentName)
|
|
.AsIChatClient();
|
|
|
|
// 2. Create specialized agents with their tools
|
|
ChatClientAgent qaEngineer = new(
|
|
client,
|
|
"You are a QA engineer responsible for running tests before deployment. Run the appropriate test suites and report results clearly.",
|
|
"QAEngineer",
|
|
"QA engineer who runs tests",
|
|
[AIFunctionFactory.Create(RunTests)]);
|
|
|
|
ChatClientAgent devopsEngineer = new(
|
|
client,
|
|
"You are a DevOps engineer responsible for deployments. First check staging status and create a rollback plan, then proceed with production deployment. Always ensure safety measures are in place before deploying.",
|
|
"DevOpsEngineer",
|
|
"DevOps engineer who handles deployments",
|
|
[
|
|
AIFunctionFactory.Create(CheckStagingStatus),
|
|
AIFunctionFactory.Create(CreateRollbackPlan),
|
|
new ApprovalRequiredAIFunction(AIFunctionFactory.Create(DeployToProduction))
|
|
]);
|
|
|
|
// 3. Create custom GroupChatManager with speaker selection logic
|
|
DeploymentGroupChatManager manager = new([qaEngineer, devopsEngineer])
|
|
{
|
|
MaximumIterationCount = 4 // Limit to 4 rounds
|
|
};
|
|
|
|
// 4. Build a group chat workflow with the custom manager
|
|
Workflow workflow = AgentWorkflowBuilder
|
|
.CreateGroupChatBuilderWith(_ => manager)
|
|
.AddParticipants(qaEngineer, devopsEngineer)
|
|
.Build();
|
|
|
|
// 5. Start the workflow
|
|
Console.WriteLine("Starting group chat workflow for software deployment...");
|
|
Console.WriteLine($"Agents: [{qaEngineer.Name}, {devopsEngineer.Name}]");
|
|
Console.WriteLine(new string('-', 60));
|
|
|
|
List<ChatMessage> messages = [new(ChatRole.User, "We need to deploy version 2.4.0 to production. Please coordinate the deployment.")];
|
|
|
|
await using StreamingRun run = await InProcessExecution.Lockstep.RunStreamingAsync(workflow, messages);
|
|
await run.TrySendMessageAsync(new TurnToken(emitEvents: true));
|
|
|
|
string? lastExecutorId = null;
|
|
await foreach (WorkflowEvent evt in run.WatchStreamAsync())
|
|
{
|
|
switch (evt)
|
|
{
|
|
case RequestInfoEvent e:
|
|
{
|
|
if (e.Request.TryGetDataAs(out ToolApprovalRequestContent? approvalRequestContent))
|
|
{
|
|
Console.WriteLine();
|
|
Console.WriteLine($"[APPROVAL REQUIRED] From agent: {e.Request.PortInfo.PortId}");
|
|
Console.WriteLine($" Tool: {((FunctionCallContent)approvalRequestContent.ToolCall).Name}");
|
|
Console.WriteLine($" Arguments: {JsonSerializer.Serialize(((FunctionCallContent)approvalRequestContent.ToolCall).Arguments)}");
|
|
Console.WriteLine();
|
|
|
|
// Approve the tool call request
|
|
Console.WriteLine($"Tool: {((FunctionCallContent)approvalRequestContent.ToolCall).Name} approved");
|
|
await run.SendResponseAsync(e.Request.CreateResponse(approvalRequestContent.CreateResponse(approved: true)));
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
case AgentResponseUpdateEvent e:
|
|
{
|
|
if (e.ExecutorId != lastExecutorId)
|
|
{
|
|
if (lastExecutorId is not null)
|
|
{
|
|
Console.WriteLine();
|
|
}
|
|
|
|
Console.WriteLine($"- {e.ExecutorId}: ");
|
|
lastExecutorId = e.ExecutorId;
|
|
}
|
|
|
|
Console.Write(e.Update.Text);
|
|
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
Console.WriteLine();
|
|
Console.WriteLine(new string('-', 60));
|
|
Console.WriteLine("Deployment workflow completed successfully!");
|
|
Console.WriteLine("All agents have finished their tasks.");
|
|
}
|
|
|
|
// Tool definitions - These are called by the agents during workflow execution
|
|
[Description("Run automated tests for the application.")]
|
|
private static string RunTests([Description("Name of the test suite to run")] string testSuite)
|
|
=> $"Test suite '{testSuite}' completed: 47 passed, 0 failed, 0 skipped";
|
|
|
|
[Description("Check the current status of the staging environment.")]
|
|
private static string CheckStagingStatus()
|
|
=> "Staging environment: Healthy, Version 2.3.0 deployed, All services running";
|
|
|
|
[Description("Deploy specified components to production. Requires human approval.")]
|
|
private static string DeployToProduction(
|
|
[Description("The version to deploy")] string version,
|
|
[Description("Comma-separated list of components to deploy")] string components)
|
|
=> $"Production deployment complete: Version {version}, Components: {components}";
|
|
|
|
[Description("Create a rollback plan for the deployment.")]
|
|
private static string CreateRollbackPlan([Description("The version being deployed")] string version)
|
|
=> $"Rollback plan created for version {version}: Automated rollback to v2.2.0 if health checks fail within 5 minutes";
|
|
}
|