mirror of
https://github.com/microsoft/agent-framework.git
synced 2026-06-16 21:04:09 +08:00
b25b0af49b
* refactor: Unify ExecutorIsh and ExecutorRegistration => ExecutorBinding * Switch to more modern Record type-tree for Sum Types * Unify APIs for getting ExecutorBinding * Fix an issue where workflows consisting entirely of cross-run shareable executors which are not instance-resettable do not properly clear state when running non-concurrently. * feat: Simplify function-to-executor pattern * refactor: Normalize API naming
398 lines
16 KiB
C#
398 lines
16 KiB
C#
// Copyright (c) Microsoft. All rights reserved.
|
|
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.IO;
|
|
using System.Linq;
|
|
using System.Text.Json;
|
|
using System.Threading.Tasks;
|
|
using FluentAssertions;
|
|
using Microsoft.Agents.AI.Workflows.Sample;
|
|
|
|
namespace Microsoft.Agents.AI.Workflows.UnitTests;
|
|
|
|
internal enum ExecutionEnvironment
|
|
{
|
|
InProcess_Lockstep,
|
|
InProcess_OffThread,
|
|
InProcess_Concurrent
|
|
}
|
|
|
|
public class SampleSmokeTest
|
|
{
|
|
[Theory]
|
|
[InlineData(ExecutionEnvironment.InProcess_Lockstep)]
|
|
[InlineData(ExecutionEnvironment.InProcess_OffThread)]
|
|
[InlineData(ExecutionEnvironment.InProcess_Concurrent)]
|
|
internal async Task Test_RunSample_Step1Async(ExecutionEnvironment environment)
|
|
{
|
|
using StringWriter writer = new();
|
|
|
|
await Step1EntryPoint.RunAsync(writer, environment.ToWorkflowExecutionEnvironment());
|
|
|
|
string result = writer.ToString();
|
|
string[] lines = result.Split([Environment.NewLine], StringSplitOptions.RemoveEmptyEntries);
|
|
|
|
const string INPUT = "Hello, World!";
|
|
|
|
Assert.Collection(lines,
|
|
line => Assert.Contains($"UppercaseExecutor: {INPUT.ToUpperInvariant()}", line),
|
|
line => Assert.Contains($"ReverseTextExecutor: {new string(INPUT.ToUpperInvariant().Reverse().ToArray())}", line)
|
|
);
|
|
}
|
|
|
|
[Theory]
|
|
[InlineData(ExecutionEnvironment.InProcess_Lockstep)]
|
|
[InlineData(ExecutionEnvironment.InProcess_OffThread)]
|
|
[InlineData(ExecutionEnvironment.InProcess_Concurrent)]
|
|
internal async Task Test_RunSample_Step1aAsync(ExecutionEnvironment environment)
|
|
{
|
|
using StringWriter writer = new();
|
|
|
|
await Step1aEntryPoint.RunAsync(writer, environment.ToWorkflowExecutionEnvironment());
|
|
|
|
string result = writer.ToString();
|
|
string[] lines = result.Split([Environment.NewLine], StringSplitOptions.RemoveEmptyEntries);
|
|
|
|
const string INPUT = "Hello, World!";
|
|
|
|
Assert.Collection(lines,
|
|
line => Assert.Contains($"UppercaseExecutor: {INPUT.ToUpperInvariant()}", line),
|
|
line => Assert.Contains($"ReverseTextExecutor: {string.Concat(INPUT.ToUpperInvariant().Reverse())}", line)
|
|
);
|
|
}
|
|
|
|
[Theory]
|
|
[InlineData(ExecutionEnvironment.InProcess_Lockstep)]
|
|
[InlineData(ExecutionEnvironment.InProcess_OffThread)]
|
|
[InlineData(ExecutionEnvironment.InProcess_Concurrent)]
|
|
internal async Task Test_RunSample_Step2Async(ExecutionEnvironment environment)
|
|
{
|
|
using StringWriter writer = new();
|
|
|
|
string spamResult = await Step2EntryPoint.RunAsync(writer, environment.ToWorkflowExecutionEnvironment());
|
|
|
|
Assert.Equal(RemoveSpamExecutor.ActionResult, spamResult);
|
|
|
|
string nonSpamResult = await Step2EntryPoint.RunAsync(writer, environment.ToWorkflowExecutionEnvironment(), "This is a valid message.");
|
|
|
|
Assert.Equal(RespondToMessageExecutor.ActionResult, nonSpamResult);
|
|
}
|
|
|
|
[Theory]
|
|
[InlineData(ExecutionEnvironment.InProcess_Lockstep)]
|
|
[InlineData(ExecutionEnvironment.InProcess_OffThread)]
|
|
[InlineData(ExecutionEnvironment.InProcess_Concurrent)]
|
|
internal async Task Test_RunSample_Step3Async(ExecutionEnvironment environment)
|
|
{
|
|
using StringWriter writer = new();
|
|
|
|
string guessResult = await Step3EntryPoint.RunAsync(writer, environment.ToWorkflowExecutionEnvironment());
|
|
|
|
Assert.Equal("Guessed the number: 42", guessResult);
|
|
}
|
|
|
|
[Theory]
|
|
[InlineData(ExecutionEnvironment.InProcess_Lockstep)]
|
|
[InlineData(ExecutionEnvironment.InProcess_OffThread)]
|
|
[InlineData(ExecutionEnvironment.InProcess_Concurrent)]
|
|
internal async Task Test_RunSample_Step4Async(ExecutionEnvironment environment)
|
|
{
|
|
using StringWriter writer = new();
|
|
|
|
VerifyingPlaybackResponder<string, int> responder = new(
|
|
("Guess the number.", 50),
|
|
("Your guess was too high. Try again.", 23),
|
|
("Your guess was too low. Try again.", 42));
|
|
|
|
string guessResult = await Step4EntryPoint.RunAsync(writer, userGuessCallback: responder.InvokeNext, environment.ToWorkflowExecutionEnvironment());
|
|
Assert.Equal("You guessed correctly! You Win!", guessResult);
|
|
}
|
|
|
|
[Theory]
|
|
[InlineData(ExecutionEnvironment.InProcess_Lockstep)]
|
|
[InlineData(ExecutionEnvironment.InProcess_OffThread)]
|
|
[InlineData(ExecutionEnvironment.InProcess_Concurrent)]
|
|
internal async Task Test_RunSample_Step5Async(ExecutionEnvironment environment)
|
|
{
|
|
using StringWriter writer = new();
|
|
|
|
VerifyingPlaybackResponder<string, int> responder = new(
|
|
// Iteration 1
|
|
("Guess the number.", 50),
|
|
("Your guess was too high. Try again.", 23),
|
|
|
|
// Iteration 2
|
|
("Your guess was too high. Try again.", 23),
|
|
("Your guess was too low. Try again.", 42)
|
|
);
|
|
|
|
string guessResult = await Step5EntryPoint.RunAsync(writer, userGuessCallback: responder.InvokeNext, environment.ToWorkflowExecutionEnvironment());
|
|
Assert.Equal("You guessed correctly! You Win!", guessResult);
|
|
}
|
|
|
|
[Theory]
|
|
[InlineData(ExecutionEnvironment.InProcess_Lockstep)]
|
|
[InlineData(ExecutionEnvironment.InProcess_OffThread)]
|
|
[InlineData(ExecutionEnvironment.InProcess_Concurrent)]
|
|
internal async Task Test_RunSample_Step5aAsync(ExecutionEnvironment environment)
|
|
{
|
|
using StringWriter writer = new();
|
|
|
|
VerifyingPlaybackResponder<string, int> responder = new(
|
|
// Iteration 1
|
|
("Guess the number.", 50),
|
|
("Your guess was too high. Try again.", 23),
|
|
|
|
// Iteration 2
|
|
("Your guess was too high. Try again.", 23),
|
|
("Your guess was too low. Try again.", 42)
|
|
);
|
|
|
|
string guessResult = await Step5EntryPoint.RunAsync(writer, userGuessCallback: responder.InvokeNext, environment.ToWorkflowExecutionEnvironment(), rehydrateToRestore: true);
|
|
Assert.Equal("You guessed correctly! You Win!", guessResult);
|
|
}
|
|
|
|
[Theory]
|
|
[InlineData(ExecutionEnvironment.InProcess_Lockstep)]
|
|
[InlineData(ExecutionEnvironment.InProcess_OffThread)]
|
|
[InlineData(ExecutionEnvironment.InProcess_Concurrent)]
|
|
internal async Task Test_RunSample_Step5bAsync(ExecutionEnvironment environment)
|
|
{
|
|
using StringWriter writer = new();
|
|
|
|
VerifyingPlaybackResponder<string, int> responder = new(
|
|
// Iteration 1
|
|
("Guess the number.", 50),
|
|
("Your guess was too high. Try again.", 23),
|
|
|
|
// Iteration 2
|
|
("Your guess was too high. Try again.", 23),
|
|
("Your guess was too low. Try again.", 42)
|
|
);
|
|
|
|
JsonSerializerOptions options = new(SampleJsonContext.Default.Options);
|
|
options.MakeReadOnly();
|
|
|
|
CheckpointManager memoryJsonManager = CheckpointManager.CreateJson(new InMemoryJsonStore(), options);
|
|
string guessResult = await Step5EntryPoint.RunAsync(writer, userGuessCallback: responder.InvokeNext, environment.ToWorkflowExecutionEnvironment(), rehydrateToRestore: true, checkpointManager: memoryJsonManager);
|
|
Assert.Equal("You guessed correctly! You Win!", guessResult);
|
|
}
|
|
|
|
[Theory]
|
|
[InlineData(ExecutionEnvironment.InProcess_Lockstep)]
|
|
[InlineData(ExecutionEnvironment.InProcess_OffThread)]
|
|
[InlineData(ExecutionEnvironment.InProcess_Concurrent)]
|
|
internal async Task Test_RunSample_Step6Async(ExecutionEnvironment environment)
|
|
{
|
|
using StringWriter writer = new();
|
|
|
|
await Step6EntryPoint.RunAsync(writer, environment.ToWorkflowExecutionEnvironment());
|
|
|
|
string result = writer.ToString();
|
|
string[] lines = result.Split([Environment.NewLine], StringSplitOptions.RemoveEmptyEntries);
|
|
|
|
Assert.Collection(lines,
|
|
line => Assert.Contains($"{HelloAgent.DefaultId}: {HelloAgent.Greeting}", line),
|
|
line => Assert.Contains($"{Step6EntryPoint.EchoAgentId}: {Step6EntryPoint.EchoPrefix}{HelloAgent.Greeting}", line)
|
|
);
|
|
}
|
|
|
|
[Theory]
|
|
[InlineData(ExecutionEnvironment.InProcess_Lockstep)]
|
|
[InlineData(ExecutionEnvironment.InProcess_OffThread)]
|
|
[InlineData(ExecutionEnvironment.InProcess_Concurrent)]
|
|
internal async Task Test_RunSample_Step7Async(ExecutionEnvironment environment)
|
|
{
|
|
using StringWriter writer = new();
|
|
|
|
await Step7EntryPoint.RunAsync(writer, environment.ToWorkflowExecutionEnvironment());
|
|
|
|
string result = writer.ToString();
|
|
string[] lines = result.Split([Environment.NewLine], StringSplitOptions.RemoveEmptyEntries);
|
|
|
|
Assert.Collection(lines,
|
|
line => Assert.Contains($"{HelloAgent.DefaultId}: {HelloAgent.Greeting}", line),
|
|
line => Assert.Contains($"{Step7EntryPoint.EchoAgentId}: {Step7EntryPoint.EchoPrefix}{HelloAgent.Greeting}", line),
|
|
line => Assert.Contains($"{HelloAgent.DefaultId}: {HelloAgent.Greeting}", line),
|
|
line => Assert.Contains($"{Step7EntryPoint.EchoAgentId}: {Step7EntryPoint.EchoPrefix}{HelloAgent.Greeting}", line)
|
|
);
|
|
}
|
|
|
|
[Theory]
|
|
[InlineData(ExecutionEnvironment.InProcess_Lockstep)]
|
|
[InlineData(ExecutionEnvironment.InProcess_OffThread)]
|
|
[InlineData(ExecutionEnvironment.InProcess_Concurrent)]
|
|
internal async Task Test_RunSample_Step8Async(ExecutionEnvironment environment)
|
|
{
|
|
List<string> textsToProcess = [
|
|
"Hello world! This is a simple test.",
|
|
"Python is a powerful programming language used for many applications.",
|
|
"Short text.",
|
|
"This is a longer text with multiple sentences. It contains more words and characters. We use it to test our text processing workflow.",
|
|
"",
|
|
" Spaces around text ",
|
|
];
|
|
|
|
using StringWriter writer = new();
|
|
|
|
List<TextProcessingResult> results = await Step8EntryPoint.RunAsync(writer, environment.ToWorkflowExecutionEnvironment(), textsToProcess);
|
|
Assert.Equal(textsToProcess.Count, results.Count);
|
|
|
|
Assert.Collection(results,
|
|
textsToProcess.Select(CreateValidator).ToArray());
|
|
|
|
Action<TextProcessingResult> CreateValidator(string textToProcess, int index)
|
|
{
|
|
return result =>
|
|
{
|
|
TextProcessingResult expected = new(
|
|
TaskId: $"Task{index}",
|
|
Text: textToProcess,
|
|
WordCount: textToProcess.Split([' '], StringSplitOptions.RemoveEmptyEntries).Length,
|
|
ChatCount: textToProcess.Length
|
|
);
|
|
|
|
result.Should().Be(expected);
|
|
};
|
|
}
|
|
}
|
|
|
|
[Theory]
|
|
[InlineData(ExecutionEnvironment.InProcess_Lockstep)]
|
|
[InlineData(ExecutionEnvironment.InProcess_OffThread)]
|
|
[InlineData(ExecutionEnvironment.InProcess_Concurrent)]
|
|
internal async Task Test_RunSample_Step9Async(ExecutionEnvironment environment)
|
|
{
|
|
using StringWriter writer = new();
|
|
_ = await Step9EntryPoint.RunAsync(writer, environment.ToWorkflowExecutionEnvironment());
|
|
}
|
|
|
|
[Theory]
|
|
[InlineData(ExecutionEnvironment.InProcess_Lockstep)]
|
|
[InlineData(ExecutionEnvironment.InProcess_OffThread)]
|
|
[InlineData(ExecutionEnvironment.InProcess_Concurrent)]
|
|
internal async Task Test_RunSample_Step10Async(ExecutionEnvironment environment)
|
|
{
|
|
List<string> inputs = ["1", "2", "3"];
|
|
|
|
using StringWriter writer = new();
|
|
await Step10EntryPoint.RunAsync(writer, environment.ToWorkflowExecutionEnvironment(), inputs);
|
|
|
|
string[] lines = writer.ToString().Split(['\r', '\n'], StringSplitOptions.RemoveEmptyEntries);
|
|
Assert.Collection(lines,
|
|
inputs.Select(CreateValidator).ToArray());
|
|
|
|
Action<string> CreateValidator(string expected) => actual => actual.Should().Be($"Echo: {expected}");
|
|
}
|
|
|
|
[Theory]
|
|
[InlineData(ExecutionEnvironment.InProcess_Lockstep)]
|
|
[InlineData(ExecutionEnvironment.InProcess_OffThread)]
|
|
[InlineData(ExecutionEnvironment.InProcess_Concurrent)]
|
|
internal async Task Test_RunSample_Step11Async(ExecutionEnvironment environment)
|
|
{
|
|
List<string> inputs = ["1", "2", "3"];
|
|
|
|
using StringWriter writer = new();
|
|
await Step11EntryPoint.RunAsync(writer, environment.ToWorkflowExecutionEnvironment(), inputs);
|
|
|
|
string[] lines = writer.ToString().Split(['\r', '\n'], StringSplitOptions.RemoveEmptyEntries);
|
|
|
|
Array.Sort(lines, StringComparer.OrdinalIgnoreCase);
|
|
|
|
string[] expected = Enumerable.Range(1, Step11EntryPoint.AgentCount)
|
|
.SelectMany(agentNumber => inputs.Select(input => Step11EntryPoint.ExpectedOutputForInput(input, agentNumber)))
|
|
.ToArray();
|
|
|
|
Array.Sort(expected, StringComparer.OrdinalIgnoreCase);
|
|
|
|
Assert.Collection(lines,
|
|
expected.Select(CreateValidator).ToArray());
|
|
|
|
Action<string> CreateValidator(string expected) => actual => actual.Should().Be(expected);
|
|
}
|
|
|
|
[Theory]
|
|
[InlineData(ExecutionEnvironment.InProcess_Lockstep)]
|
|
[InlineData(ExecutionEnvironment.InProcess_OffThread)]
|
|
[InlineData(ExecutionEnvironment.InProcess_Concurrent)]
|
|
internal async Task Test_RunSample_Step12Async(ExecutionEnvironment environment)
|
|
{
|
|
List<string> inputs = ["1", "2", "3"];
|
|
|
|
using StringWriter writer = new();
|
|
await Step12EntryPoint.RunAsync(writer, environment.ToWorkflowExecutionEnvironment(), inputs);
|
|
|
|
string[] lines = writer.ToString().Split(['\r', '\n'], StringSplitOptions.RemoveEmptyEntries);
|
|
|
|
// The expectation is that each agent will echo each input along with every echo from previous agents
|
|
// E.g.:
|
|
// (user): 1
|
|
// (a1): 1:1
|
|
// (a2): 2:1
|
|
// (a2): 2:1:1
|
|
|
|
// If there were three agents, it would then be followed by:
|
|
// (a3): 3:1
|
|
// (a3): 3:1:1
|
|
// (a3): 3:2:1
|
|
// (a3): 3:2:1:1
|
|
|
|
string[] expected = inputs.SelectMany(input => EchoesForInput(input)).ToArray();
|
|
|
|
Console.Error.WriteLine("Expected lines: ");
|
|
foreach (string expectedLine in expected)
|
|
{
|
|
Console.Error.WriteLine($"\t{expectedLine}");
|
|
}
|
|
|
|
Console.Error.WriteLine("Actual lines: ");
|
|
foreach (string line in lines)
|
|
{
|
|
Console.Error.WriteLine($"\t{line}");
|
|
}
|
|
|
|
Assert.Collection(lines,
|
|
expected.Select(CreateValidator).ToArray());
|
|
|
|
IEnumerable<string> EchoesForInput(string input)
|
|
{
|
|
List<string> echoes = [$"{Step12EntryPoint.EchoPrefixForAgent(1)}{input}"];
|
|
for (int i = 2; i <= Step12EntryPoint.AgentCount; i++)
|
|
{
|
|
string agentPrefix = Step12EntryPoint.EchoPrefixForAgent(i);
|
|
List<string> newEchoes = [$"{agentPrefix}{input}", .. echoes.Select(echo => $"{agentPrefix}{echo}")];
|
|
echoes.AddRange(newEchoes);
|
|
}
|
|
|
|
return echoes;
|
|
}
|
|
|
|
Action<string> CreateValidator(string expected) => actual => actual.Should().Be(expected);
|
|
}
|
|
}
|
|
|
|
internal sealed class VerifyingPlaybackResponder<TInput, TResponse>
|
|
{
|
|
public (TInput input, TResponse response)[] Responses { get; }
|
|
private int _position;
|
|
|
|
public VerifyingPlaybackResponder(params (TInput input, TResponse response)[] responses)
|
|
{
|
|
this.Responses = responses;
|
|
}
|
|
|
|
public int Remaining => Math.Max(0, this.Responses.Length - this._position);
|
|
|
|
public TResponse InvokeNext(TInput input)
|
|
{
|
|
Assert.True(this.Remaining > 0);
|
|
|
|
(TInput expectedInput, TResponse expectedResponse) = this.Responses[this._position++];
|
|
Assert.Equal(expectedInput, input);
|
|
|
|
return expectedResponse;
|
|
}
|
|
}
|