Files
agent-framework/dotnet/tests/Microsoft.Agents.AI.Hosting.AzureFunctions.UnitTests/DurableAgentFunctionMetadataTransformerTests.cs
Dmytro Struk 67a8147151 .NET: Python: Azure Functions feature branch (#1916)
* 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>
2025-11-13 02:00:53 +00:00

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);
}
}