Files
agent-framework/dotnet/tests/Microsoft.Agents.AI.Workflows.UnitTests/ChatProtocolExecutorTests.cs
Jacob Alber 6a3d22598f .NET: [BREAKING] Implement Polymorphic Routing (#3792)
* feat: Implement Polymorphic Routing

* feat: Add support for Send/Yield annotations with basic Executor

* Adds annotations to Declarative workflow executors

* fix: Address PR Comments

* Implicit filter in collection loops
* Remove debug / usused / superfluous code
* Fix ProtocolBuilder implicit output registrations
* Fix logic error in ExecuteRouteGeneratorTests.ClassWithManualConfigureProtocol_DoesNotGenerate

* fix: Solidify type checks and send/yield type registrations

* fix: Suppress generation of TurnTokens out of AggregateTurnMessagesExecutor

* Fixes an issue where ConcurrentEndExecutor is not expecting TurnTokens.

* fix: Add ProtocolBuilder support for chained-delegation

* Updates Declarative pacakge to rely on chained-delegation Send/Yield registration
* Renames DeclarativeActionExectuor's new ExecuteAsync to ExecuteActionAsync to avoid colliding with Executor.ExecutoeAsync

* fix: Address PR Comments

* Fixes type mapping in FanInEdgeRunner
* Fixes and expalins send/yield type registration in FunctionExecutor

* fixup: build-break

* fix: Add missing SendsMesage declaration to InvokeAzureAgentExecutor
2026-02-19 14:09:03 +00:00

244 lines
9.5 KiB
C#

// Copyright (c) Microsoft. All rights reserved.
using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading;
using System.Threading.Tasks;
using FluentAssertions;
using Microsoft.Agents.AI.Workflows.Checkpointing;
using Microsoft.Extensions.AI;
namespace Microsoft.Agents.AI.Workflows.UnitTests;
/// <summary>
/// Tests for <see cref="ChatProtocolExecutor"/> to verify message routing behavior.
/// </summary>
public class ChatProtocolExecutorTests
{
private sealed class TestChatProtocolExecutor : ChatProtocolExecutor
{
public List<ChatMessage> ReceivedMessages { get; } = [];
public int TurnCount { get; private set; }
public TestChatProtocolExecutor(string id = "test-executor", ChatProtocolExecutorOptions? options = null)
: base(id, options)
{
}
protected override async ValueTask TakeTurnAsync(
List<ChatMessage> messages,
IWorkflowContext context,
bool? emitEvents,
CancellationToken cancellationToken = default)
{
this.ReceivedMessages.AddRange(messages);
this.TurnCount++;
// Send messages back to context so they can be collected
await context.SendMessageAsync(messages, cancellationToken: cancellationToken);
}
}
[Fact]
public void ChatProtocolExecutor_DescribedProtocol_IsChatProtocol()
{
// Arrange
TestChatProtocolExecutor executor = new();
ProtocolDescriptor protocol = executor.DescribeProtocol();
// Act & Assert
protocol.Should().Match<ProtocolDescriptor>(protocol => protocol.IsChatProtocol());
}
[Fact]
public async Task ChatProtocolExecutor_Handles_ListOfChatMessagesAsync()
{
// Arrange
TestChatProtocolExecutor executor = new();
TestWorkflowContext context = new(executor.Id);
List<ChatMessage> messages =
[
new ChatMessage(ChatRole.User, "Hello"),
new ChatMessage(ChatRole.User, "World")
];
// Act - Send List<ChatMessage> via ExecuteAsync
await executor.ExecuteCoreAsync(messages, new TypeId(typeof(List<ChatMessage>)), context);
await executor.TakeTurnAsync(new TurnToken(emitEvents: false), context);
// Assert
executor.ReceivedMessages.Should().HaveCount(2);
executor.ReceivedMessages[0].Text.Should().Be("Hello");
executor.ReceivedMessages[1].Text.Should().Be("World");
executor.TurnCount.Should().Be(1);
}
[Fact]
public async Task ChatProtocolExecutor_Handles_ArrayOfChatMessagesAsync()
{
// Arrange
TestChatProtocolExecutor executor = new();
TestWorkflowContext context = new(executor.Id);
ChatMessage[] messages =
[
new ChatMessage(ChatRole.System, "System message"),
new ChatMessage(ChatRole.User, "User query"),
new ChatMessage(ChatRole.Assistant, "Agent reply")
];
// Act - Send as ChatMessage[]
await executor.ExecuteCoreAsync(messages, new TypeId(typeof(ChatMessage[])), context);
await executor.TakeTurnAsync(new TurnToken(emitEvents: false), context);
// Assert
executor.ReceivedMessages.Should().HaveCount(3);
executor.ReceivedMessages[0].Role.Should().Be(ChatRole.System);
executor.ReceivedMessages[1].Role.Should().Be(ChatRole.User);
executor.ReceivedMessages[2].Role.Should().Be(ChatRole.Assistant);
executor.TurnCount.Should().Be(1);
}
[Fact]
public async Task ChatProtocolExecutor_Handles_SingleChatMessageAsync()
{
// Arrange
TestChatProtocolExecutor executor = new();
TestWorkflowContext context = new(executor.Id);
var message = new ChatMessage(ChatRole.User, "Single message");
// Act - Send as single ChatMessage
await executor.ExecuteCoreAsync(message, new TypeId(typeof(ChatMessage)), context);
await executor.TakeTurnAsync(new TurnToken(emitEvents: false), context);
// Assert
executor.ReceivedMessages.Should().HaveCount(1);
executor.ReceivedMessages[0].Text.Should().Be("Single message");
executor.TurnCount.Should().Be(1);
}
[Fact]
public async Task ChatProtocolExecutor_AccumulatesAndClearsMessagesPerTurnAsync()
{
TestChatProtocolExecutor executor = new();
TestWorkflowContext context = new(executor.Id);
// Send multiple message batches before taking a turn
await executor.ExecuteCoreAsync(new ChatMessage(ChatRole.User, "Message 1"), new TypeId(typeof(ChatMessage)), context);
await executor.ExecuteCoreAsync(new List<ChatMessage>
{
new(ChatRole.User, "Message 2"),
new(ChatRole.User, "Message 3")
}, new TypeId(typeof(List<ChatMessage>)), context);
await executor.ExecuteCoreAsync(new ChatMessage[] { new(ChatRole.User, "Message 4") }, new TypeId(typeof(ChatMessage[])), context);
await executor.TakeTurnAsync(new TurnToken(emitEvents: false), context);
executor.ReceivedMessages.Should().HaveCount(4);
executor.ReceivedMessages.Select(m => m.Text).Should().Equal("Message 1", "Message 2", "Message 3", "Message 4");
executor.TurnCount.Should().Be(1);
executor.ReceivedMessages.Clear();
// Second turn should process new messages only
await executor.ExecuteCoreAsync(new List<ChatMessage>
{
new(ChatRole.User, "Second batch")
}, new TypeId(typeof(List<ChatMessage>)), context);
await executor.TakeTurnAsync(new TurnToken(emitEvents: false), context);
executor.ReceivedMessages.Should().HaveCount(1);
executor.ReceivedMessages[0].Text.Should().Be("Second batch");
executor.TurnCount.Should().Be(2);
}
[Fact]
public async Task ChatProtocolExecutor_WithStringRole_ConvertsStringToMessageAsync()
{
TestChatProtocolExecutor executor = new(
options: new ChatProtocolExecutorOptions
{
StringMessageChatRole = ChatRole.User
});
TestWorkflowContext context = new(executor.Id);
await executor.ExecuteCoreAsync("String message", new TypeId(typeof(string)), context);
await executor.TakeTurnAsync(new TurnToken(emitEvents: false), context);
executor.ReceivedMessages.Should().HaveCount(1);
executor.ReceivedMessages[0].Role.Should().Be(ChatRole.User);
executor.ReceivedMessages[0].Text.Should().Be("String message");
}
[Fact]
public async Task ChatProtocolExecutor_EmptyCollection_HandledCorrectlyAsync()
{
TestChatProtocolExecutor executor = new();
TestWorkflowContext context = new(executor.Id);
await executor.ExecuteCoreAsync(new List<ChatMessage>(), new TypeId(typeof(List<ChatMessage>)), context);
await executor.ExecuteCoreAsync(Array.Empty<ChatMessage>(), new TypeId(typeof(ChatMessage[])), context);
await executor.TakeTurnAsync(new TurnToken(emitEvents: false), context);
executor.ReceivedMessages.Should().BeEmpty();
executor.TurnCount.Should().Be(1);
}
[Theory]
[InlineData(typeof(List<ChatMessage>))]
[InlineData(typeof(ChatMessage[]))]
public async Task ChatProtocolExecutor_RoutesCollectionTypesAsync(Type collectionType)
{
TestChatProtocolExecutor executor = new();
TestWorkflowContext context = new(executor.Id);
var sourceMessages = new[] { new ChatMessage(ChatRole.User, "Test message") };
object messagesToSend = collectionType == typeof(List<ChatMessage>) ? sourceMessages.ToList() : sourceMessages;
await executor.ExecuteCoreAsync(messagesToSend, new TypeId(collectionType), context);
await executor.TakeTurnAsync(new TurnToken(emitEvents: false), context);
executor.ReceivedMessages.Should().HaveCount(1);
executor.ReceivedMessages[0].Text.Should().Be("Test message");
}
[Fact]
public async Task ChatProtocolExecutor_MultipleTurns_EachTurnProcessesSeparatelyAsync()
{
TestChatProtocolExecutor executor = new();
TestWorkflowContext context = new(executor.Id);
await executor.ExecuteCoreAsync(new List<ChatMessage> { new(ChatRole.User, "Turn 1") }, new TypeId(typeof(List<ChatMessage>)), context);
await executor.TakeTurnAsync(new TurnToken(emitEvents: false), context);
executor.ReceivedMessages.Should().HaveCount(1);
await executor.ExecuteCoreAsync(new ChatMessage(ChatRole.User, "Turn 2"), new TypeId(typeof(ChatMessage)), context);
await executor.TakeTurnAsync(new TurnToken(emitEvents: false), context);
executor.ReceivedMessages.Should().HaveCount(2);
executor.ReceivedMessages[0].Text.Should().Be("Turn 1");
executor.ReceivedMessages[1].Text.Should().Be("Turn 2");
executor.TurnCount.Should().Be(2);
}
[Fact]
public async Task ChatProtocolExecutor_InitialWorkflowMessages_RoutedCorrectlyAsync()
{
TestChatProtocolExecutor executor = new();
TestWorkflowContext context = new(executor.Id);
List<ChatMessage> initialMessages = [new ChatMessage(ChatRole.User, "Kick off the workflow")];
await executor.ExecuteCoreAsync(initialMessages, new TypeId(typeof(List<ChatMessage>)), context);
await executor.TakeTurnAsync(new TurnToken(emitEvents: false), context);
executor.ReceivedMessages.Should().NotBeEmpty();
executor.ReceivedMessages.Should().HaveCount(1);
executor.ReceivedMessages[0].Text.Should().Be("Kick off the workflow");
}
}