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>
187 lines
7.7 KiB
C#
187 lines
7.7 KiB
C#
// 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<string, Func<IServiceProvider, AIAgent>> 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<string, FunctionsAgentOptions>
|
|
{
|
|
{ "testAgent", options }
|
|
});
|
|
|
|
List<IFunctionMetadata> metadataList = BuildFunctionMetadataList(initialMetadataEntryCount);
|
|
|
|
DurableAgentFunctionMetadataTransformer transformer = new(
|
|
agents,
|
|
NullLogger<DurableAgentFunctionMetadataTransformer>.Instance,
|
|
new FakeServiceProvider(),
|
|
agentOptionsProvider);
|
|
|
|
// Act
|
|
transformer.Transform(metadataList);
|
|
|
|
// Assert
|
|
Assert.Equal(initialMetadataEntryCount + expectedMetadataCount, metadataList.Count);
|
|
|
|
DefaultFunctionMetadata agentTrigger = Assert.IsType<DefaultFunctionMetadata>(metadataList[initialMetadataEntryCount]);
|
|
Assert.Equal("dafx-testAgent", agentTrigger.Name);
|
|
Assert.Contains("entityTrigger", agentTrigger.RawBindings![0]);
|
|
|
|
if (enableHttp)
|
|
{
|
|
DefaultFunctionMetadata httpTrigger = Assert.IsType<DefaultFunctionMetadata>(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<DefaultFunctionMetadata>(metadataList[mcpIndex]);
|
|
Assert.Equal("mcptool-testAgent", mcpToolTrigger.Name);
|
|
Assert.Contains("mcpToolTrigger", mcpToolTrigger.RawBindings![0]);
|
|
}
|
|
}
|
|
|
|
[Fact]
|
|
public void Transform_AddsTriggers_ForMultipleAgents()
|
|
{
|
|
// Arrange
|
|
Dictionary<string, Func<IServiceProvider, AIAgent>> 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<string, FunctionsAgentOptions> functionsAgentOptions = new()
|
|
{
|
|
{ "agentA", agentOptionsA },
|
|
{ "agentB", agentOptionsB },
|
|
{ "agentC", agentOptionsC }
|
|
};
|
|
|
|
IFunctionsAgentOptionsProvider agentOptionsProvider = new FakeOptionsProvider(functionsAgentOptions);
|
|
DurableAgentFunctionMetadataTransformer transformer = new(
|
|
agents,
|
|
NullLogger<DurableAgentFunctionMetadataTransformer>.Instance,
|
|
new FakeServiceProvider(),
|
|
agentOptionsProvider);
|
|
|
|
const int InitialMetadataEntryCount = 2;
|
|
List<IFunctionMetadata> 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<DefaultFunctionMetadata>(
|
|
Assert.Single(metadataList, m => m.Name == $"dafx-{agentName}"));
|
|
Assert.NotNull(entityMeta.RawBindings);
|
|
Assert.Contains("entityTrigger", entityMeta.RawBindings[0]);
|
|
|
|
DefaultFunctionMetadata httpMeta =
|
|
Assert.IsType<DefaultFunctionMetadata>(
|
|
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 == "agentB" || agentName == "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<IFunctionMetadata> BuildFunctionMetadataList(int numberOfFunctions)
|
|
{
|
|
List<IFunctionMetadata> 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<string, FunctionsAgentOptions> _map;
|
|
|
|
public FakeOptionsProvider(Dictionary<string, FunctionsAgentOptions> 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);
|
|
}
|
|
}
|