mirror of
https://github.com/microsoft/agent-framework.git
synced 2026-06-16 21:04:09 +08:00
67a8147151
* Python: Add Scaffolding for Durable AzureFunctions package to Agent Framework (#1823) * Add scafolding * update readme * add code owners and label * update owners * .NET: Durable extension: initial src and unit tests (#1900) * Python: Add Durable Agent Wrapper code (#1913) * add initial changes * Move code and add single sample * Update logger * Remove unused code * address PR comments * cleanup code and address comments --------- Co-authored-by: Dmytro Struk <13853051+dmytrostruk@users.noreply.github.com> * Azure Functions .NET samples (#1939) * Python: Add Unit tests for Azurefunctions package (#1976) * Add Unit tests for Azurefunctions * remove duplicate import * .NET: [Feature Branch] Migrate state schema updates and support for agents as MCP tools (#1979) * Python: Add more samples for Azure Functions (#1980) * Move all samples * fix comments * remove dead lines * Make samples simpler * .NET: [Feature Branch] Durable Task extension integration tests (#2017) * .NET: [Feature Branch] Update OpenAI config for integration tests (#2063) * Python: Add Integration tests for AzureFunctions (#2020) * Add Integration tests * Remove DTS extension * Apply suggestions from code review Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> * Apply suggestions from code review Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> * Add pyi file for type safety * Add samples in readme * Updated all readme instructions * Address comments * Update readmes * Fix requirements * Address comments --------- Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> * .NET: [Feature Branch] Update dotnet-build-and-test.yml to support integration tests (#2070) Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> * Fix DTS startup issue and improve logging (#2103) * .NET: [Feature Branch] Introduce Azure OpenAI config for .NET pipeline (#2106) Also fixes an issue where we were trying to start docker containers for integration tests on Windows, which doesn't work. Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> * Fix uv.lock after merge * Python: Add README for Azure Functions samples setup (#2100) * Add README for Azure Functions samples setup Added setup instructions for Azure Functions samples, including environment setup, virtual environment creation, and running samples. * Update python/samples/getting_started/azure_functions/README.md Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> * Apply suggestions from code review Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> * Apply suggestion from @Copilot Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> * Apply suggestions from code review Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> --------- Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> Co-authored-by: Laveesh Rohra <larohra@microsoft.com> * Fix or remove broken markdown file links (#2115) * .NET: [Feature Branch] Update HTTP API to be consistent across languages (#2118) * Python: Fix AzureFunctions Integration Tests (#2116) * Add Identity Auth to samples * Update python/samples/getting_started/azure_functions/README.md Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> * Update python/samples/getting_started/azure_functions/01_single_agent/function_app.py Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> * Update python/samples/getting_started/azure_functions/02_multi_agent/function_app.py Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> * Update python/samples/getting_started/azure_functions/06_multi_agent_orchestration_conditionals/README.md Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> --------- Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> * Python: Fix Http Schema (#2112) * Rename to threadid * Respond in plain text * Make snake-case * Add http prefix * rename to wait-for-response * Add query param check * address comments * .NET: Remove IsPackable=false in preparation for nuget release (#2142) * Python: Move `azurefunctions` to `azure` for import (#2141) * Move import to Azure * fix mypy * Update python/packages/azurefunctions/README.md Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> * Add missing types * Address comments --------- Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> * Update python/packages/azurefunctions/pyproject.toml Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> * Update python/packages/azurefunctions/agent_framework_azurefunctions/__init__.py Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> * Fix imports * Address PR feedback from westey-m (#2150) - Adds a link from the /dotnet/samples/README.md to /dotnet/samples/AzureFunctions - Make DurableAgentThread deserialization internal for future-proofing - Update JSON serialization logic to address recently discovered issues with source generator serialization * Address comments (#2160) --------- Co-authored-by: Laveesh Rohra <larohra@microsoft.com> Co-authored-by: Chris Gillum <cgillum@microsoft.com> Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> Co-authored-by: Anirudh Garg <anirudhg@microsoft.com>
152 lines
5.7 KiB
C#
152 lines
5.7 KiB
C#
// 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<object> RunOrchestrationAsync(
|
|
[OrchestrationTrigger] TaskOrchestrationContext context)
|
|
{
|
|
// Get the input from the orchestration
|
|
ContentGenerationInput input = context.GetInput<ContentGenerationInput>()
|
|
?? 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<GeneratedContent> writerResponse = await writerAgent.RunAsync<GeneratedContent>(
|
|
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<HumanApprovalResponse>(
|
|
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<GeneratedContent>(
|
|
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);
|
|
}
|
|
}
|