Files
Peter Ibekwe aa2ff672fb .NET: Fix to emit WorkflowStartedEvent during workflow execution (#4514)
* Fix bug to emit WorkflowStartedEvent during workflow execution

* Updated based on PR comments
2026-03-12 15:45:17 +00:00

159 lines
5.7 KiB
C#

// Copyright (c) Microsoft. All rights reserved.
using System.Collections.Generic;
using System.Linq;
using System.Threading.Tasks;
using FluentAssertions;
using Microsoft.Extensions.AI;
namespace Microsoft.Agents.AI.Workflows.UnitTests;
public class AgentEventsTests
{
/// <summary>
/// Regression test for https://github.com/microsoft/agent-framework/issues/2938
/// Verifies that WorkflowOutputEvent is triggered for agent workflows built with
/// WorkflowBuilder directly (without using AgentWorkflowBuilder helpers).
/// </summary>
[Fact]
public async Task WorkflowBuilder_WithAgents_EmitsWorkflowOutputEventAsync()
{
// Arrange - Build workflow using WorkflowBuilder directly (not AgentWorkflowBuilder.BuildSequential)
AIAgent agent1 = new TestEchoAgent("agent1");
AIAgent agent2 = new TestEchoAgent("agent2");
Workflow workflow = new WorkflowBuilder(agent1)
.AddEdge(agent1, agent2)
.Build();
// Act
await using StreamingRun run = await InProcessExecution.RunStreamingAsync(workflow, new List<ChatMessage> { new(ChatRole.User, "Hello") });
await run.TrySendMessageAsync(new TurnToken(emitEvents: true));
List<WorkflowOutputEvent> outputEvents = new();
List<AgentResponseUpdateEvent> updateEvents = new();
await foreach (WorkflowEvent evt in run.WatchStreamAsync())
{
if (evt is AgentResponseUpdateEvent updateEvt)
{
updateEvents.Add(updateEvt);
}
if (evt is WorkflowOutputEvent outputEvt)
{
outputEvents.Add(outputEvt);
}
}
// Assert - AgentResponseUpdateEvent should now be a WorkflowOutputEvent
Assert.NotEmpty(updateEvents);
Assert.NotEmpty(outputEvents);
// All update events should also be output events (since AgentResponseUpdateEvent now inherits from WorkflowOutputEvent)
Assert.All(updateEvents, updateEvt => Assert.Contains(updateEvt, outputEvents));
}
/// <summary>
/// Verifies that AgentResponseUpdateEvent inherits from WorkflowOutputEvent.
/// </summary>
[Fact]
public void AgentResponseUpdateEvent_IsWorkflowOutputEvent()
{
// Arrange
AgentResponseUpdate update = new(ChatRole.Assistant, "test");
// Act
AgentResponseUpdateEvent evt = new("executor1", update);
// Assert
Assert.IsAssignableFrom<WorkflowOutputEvent>(evt);
Assert.Equal("executor1", evt.ExecutorId);
Assert.Same(update, evt.Update);
Assert.Same(update, evt.Data);
}
/// <summary>
/// Verifies that AgentResponseEvent inherits from WorkflowOutputEvent.
/// </summary>
[Fact]
public void AgentResponseEvent_IsWorkflowOutputEvent()
{
// Arrange
AgentResponse response = new(new List<ChatMessage> { new(ChatRole.Assistant, "test") });
// Act
AgentResponseEvent evt = new("executor1", response);
// Assert
Assert.IsAssignableFrom<WorkflowOutputEvent>(evt);
Assert.Equal("executor1", evt.ExecutorId);
Assert.Same(response, evt.Response);
Assert.Same(response, evt.Data);
}
/// <summary>
/// Verifies that WorkflowStartedEvent is emitted first before any SuperStepStartedEvent.
/// </summary>
[Fact]
public async Task StreamingRun_WorkflowStartedEvent_ShouldBeEmittedBefore_SuperStepStartedAsync()
{
// Arrange
TestEchoAgent agent = new("test-agent");
Workflow workflow = AgentWorkflowBuilder.BuildSequential(agent);
ChatMessage inputMessage = new(ChatRole.User, "Hello");
// Act
await using StreamingRun run = await InProcessExecution.RunStreamingAsync(workflow, new List<ChatMessage> { inputMessage });
await run.TrySendMessageAsync(new TurnToken(emitEvents: true));
List<WorkflowEvent> events = [];
await foreach (WorkflowEvent evt in run.WatchStreamAsync())
{
events.Add(evt);
}
// Assert
events.Should().NotBeEmpty();
List<WorkflowStartedEvent> startedEvents = events.OfType<WorkflowStartedEvent>().ToList();
startedEvents.Should().NotBeEmpty();
WorkflowStartedEvent? firstStartedEvent = startedEvents.FirstOrDefault();
SuperStepStartedEvent? firstSuperStepEvent = events.OfType<SuperStepStartedEvent>().FirstOrDefault();
firstSuperStepEvent.Should().NotBeNull();
int startedIndex = events.IndexOf(firstStartedEvent!);
int superStepIndex = events.IndexOf(firstSuperStepEvent!);
startedIndex.Should().BeLessThan(superStepIndex);
}
/// <summary>
/// Verifies that WorkflowStartedEvent is emitted using Lockstep execution mode.
/// </summary>
[Fact]
public async Task StreamingRun_LockstepExecution_ShouldEmit_WorkflowStartedEventAsync()
{
// Arrange
TestEchoAgent agent = new("test-agent");
Workflow workflow = AgentWorkflowBuilder.BuildSequential(agent);
ChatMessage inputMessage = new(ChatRole.User, "Hello");
// Act: Use Lockstep execution mode
await using StreamingRun run = await InProcessExecution.Lockstep.RunStreamingAsync(workflow, new List<ChatMessage> { inputMessage });
await run.TrySendMessageAsync(new TurnToken(emitEvents: true));
List<WorkflowEvent> events = [];
await foreach (WorkflowEvent evt in run.WatchStreamAsync())
{
events.Add(evt);
}
// Assert
events.Should().NotBeEmpty();
List<WorkflowStartedEvent> startedEvents = events.OfType<WorkflowStartedEvent>().ToList();
startedEvents.Should().NotBeEmpty();
}
}