mirror of
https://github.com/microsoft/agent-framework.git
synced 2026-06-16 21:04:09 +08:00
6e8c7c42c8
* refactor: Rename AggregateTurnMessagesExecutor * feat: Rework Agent Hosting for Configurability and HIL support * Adds support for selecting whether updates and/or full responses are emitted to events * Adds support for HIL/FunctionCalls (including interception) * Implements internal support for ExternalRequests from any executor (not just RequestPort) * test: Add tests for new AIAgentHostExecutor functionality * feat: Unify non-Handoff Agent Hosting * doc: More explicit documentation for `overwrite` in RouteBuilder
126 lines
4.8 KiB
C#
126 lines
4.8 KiB
C#
// Copyright (c) Microsoft. All rights reserved.
|
|
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using System.Threading.Tasks;
|
|
using FluentAssertions;
|
|
using Microsoft.Agents.AI.Workflows.Checkpointing;
|
|
using Microsoft.Agents.AI.Workflows.Execution;
|
|
using Microsoft.Agents.AI.Workflows.Specialized;
|
|
|
|
namespace Microsoft.Agents.AI.Workflows.UnitTests;
|
|
|
|
public class EdgeMapSmokeTests
|
|
{
|
|
[Fact]
|
|
public async Task Test_EdgeMap_RoutesStaticPortAsync()
|
|
{
|
|
TestRunContext runContext = new();
|
|
|
|
RequestPort staticPort = RequestPort.Create<string, int>("port1");
|
|
RequestInfoExecutor executor = new(staticPort);
|
|
EdgeMap edgeMap = new(runContext, [], [staticPort], executor.Id, null);
|
|
|
|
runContext.ConfigureExecutor(executor, edgeMap);
|
|
|
|
ExternalResponse responseMessage = new(staticPort.ToPortInfo(), "Request1", new(12));
|
|
|
|
DeliveryMapping? mapping = await edgeMap.PrepareDeliveryForResponseAsync(responseMessage);
|
|
mapping.Should().NotBeNull();
|
|
|
|
List<MessageDelivery> deliveries = mapping.Deliveries.ToList();
|
|
deliveries.Should().HaveCount(1).And.AllSatisfy(delivery => delivery.TargetId.Should().Be(executor.Id));
|
|
deliveries[0].Envelope.Message.Should().Be(responseMessage);
|
|
}
|
|
|
|
[Fact]
|
|
public async Task Test_EdgeMap_RoutesDynamicPortAsync()
|
|
{
|
|
TestRunContext runContext = new();
|
|
|
|
DynamicPortsExecutor<string, int> executor = new("executor1", "port1", "port2");
|
|
EdgeMap edgeMap = new(runContext, [], [], executor.Id, null);
|
|
|
|
runContext.ConfigureExecutor(executor, edgeMap);
|
|
|
|
await RunPortTestAsync("port1");
|
|
await RunPortTestAsync("port2");
|
|
|
|
async ValueTask RunPortTestAsync(string portId)
|
|
{
|
|
PortBinding binding = executor.PortBindings[portId];
|
|
ExternalResponse responseMessage = new(binding.Port.ToPortInfo(), $"RequestFor[{portId}]", new(10));
|
|
|
|
DeliveryMapping? mapping = await edgeMap.PrepareDeliveryForResponseAsync(responseMessage);
|
|
mapping.Should().NotBeNull();
|
|
|
|
List<MessageDelivery> deliveries = mapping.Deliveries.ToList();
|
|
deliveries.Should().HaveCount(1).And.AllSatisfy(delivery => delivery.TargetId.Should().Be(executor.Id));
|
|
deliveries[0].Envelope.Message.Should().Be(responseMessage);
|
|
}
|
|
}
|
|
|
|
[Fact]
|
|
public async Task Test_EdgeMap_DoesNotRouteUnregisteredPortAsync()
|
|
{
|
|
TestRunContext runContext = new();
|
|
|
|
RequestPort staticPort = RequestPort.Create<string, int>("port1");
|
|
RequestInfoExecutor staticExecutor = new(staticPort);
|
|
DynamicPortsExecutor<string, int> executor = new("executor1", "port2", "port3");
|
|
EdgeMap edgeMap = new(runContext, [], [staticPort], executor.Id, null);
|
|
|
|
runContext.ConfigureExecutors([staticExecutor, executor], edgeMap);
|
|
|
|
await RunPortTestAsync("port4");
|
|
|
|
async ValueTask RunPortTestAsync(string portId)
|
|
{
|
|
RequestPort fakePort = RequestPort.Create<string, int>(portId);
|
|
|
|
ExternalResponse responseMessage = new(fakePort.ToPortInfo(), $"RequestFor[{portId}]", new(10));
|
|
|
|
Func<Task<DeliveryMapping?>> mappingTask = async () => await edgeMap.PrepareDeliveryForResponseAsync(responseMessage);
|
|
await mappingTask.Should().ThrowAsync<InvalidOperationException>();
|
|
}
|
|
}
|
|
|
|
[Fact]
|
|
public async Task Test_EdgeMap_MaintainsFanInEdgeStateAsync()
|
|
{
|
|
TestRunContext runContext = new();
|
|
Dictionary<string, HashSet<Edge>> workflowEdges = [];
|
|
|
|
FanInEdgeData edgeData = new(["executor1", "executor2"], "executor3", new EdgeId(0), null);
|
|
Edge fanInEdge = new(edgeData);
|
|
|
|
workflowEdges["executor1"] = [fanInEdge];
|
|
workflowEdges["executor2"] = [fanInEdge];
|
|
EdgeMap edgeMap = new(runContext, workflowEdges, [], "executor1", null);
|
|
|
|
runContext.ConfigureExecutors(
|
|
[
|
|
new ForwardMessageExecutor<string>("executor1"),
|
|
new ForwardMessageExecutor<string>("executor2"),
|
|
new ForwardMessageExecutor<string>("executor3")
|
|
], edgeMap);
|
|
|
|
DeliveryMapping? mapping = await edgeMap.PrepareDeliveryForEdgeAsync(fanInEdge, new("part1", "executor1"));
|
|
mapping.Should().BeNull();
|
|
|
|
mapping = await edgeMap.PrepareDeliveryForEdgeAsync(fanInEdge, new("part2", "executor2"));
|
|
mapping.Should().NotBeNull();
|
|
List<MessageDelivery> deliveries = mapping.Deliveries.ToList();
|
|
|
|
deliveries.Should().HaveCount(2).And.AllSatisfy(delivery => delivery.TargetId.Should().Be("executor3"));
|
|
|
|
HashSet<string> expectedMessages = ["part1", "part2"];
|
|
foreach (MessageDelivery delivery in deliveries)
|
|
{
|
|
string message = delivery.Envelope.As<string>()!;
|
|
expectedMessages.Remove(message);
|
|
}
|
|
}
|
|
}
|