mirror of
https://github.com/microsoft/agent-framework.git
synced 2026-06-16 21:04:09 +08:00
a3a9147e61
* Rename AgentThread to AgentSession * Add more renames * Update readme files * Revert nullable variable change and further fixes. * Revert change in header name * Fix some comments and tests * Update changelog. * Address PR feedback. * Fixing code review comments. * Fix new errors after merging latest code.
376 lines
13 KiB
C#
376 lines
13 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 Moq;
|
|
using Moq.Protected;
|
|
using OpenAI.Responses;
|
|
using ChatMessage = Microsoft.Extensions.AI.ChatMessage;
|
|
using ChatRole = Microsoft.Extensions.AI.ChatRole;
|
|
using OpenAIChatMessage = OpenAI.Chat.ChatMessage;
|
|
using TextContent = Microsoft.Extensions.AI.TextContent;
|
|
|
|
namespace Microsoft.Agents.AI.OpenAI.UnitTests.Extensions;
|
|
|
|
/// <summary>
|
|
/// Unit tests for the <see cref="AIAgentWithOpenAIExtensions"/> class.
|
|
/// </summary>
|
|
public sealed class AIAgentWithOpenAIExtensionsTests
|
|
{
|
|
/// <summary>
|
|
/// Verify that RunAsync throws ArgumentNullException when agent is null.
|
|
/// </summary>
|
|
[Fact]
|
|
public async Task RunAsync_WithNullAgent_ThrowsArgumentNullExceptionAsync()
|
|
{
|
|
// Arrange
|
|
AIAgent? agent = null;
|
|
var messages = new List<OpenAIChatMessage>
|
|
{
|
|
OpenAIChatMessage.CreateUserMessage("Test message")
|
|
};
|
|
|
|
// Act & Assert
|
|
var exception = await Assert.ThrowsAsync<ArgumentNullException>(
|
|
() => agent!.RunAsync(messages));
|
|
|
|
Assert.Equal("agent", exception.ParamName);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Verify that RunAsync throws ArgumentNullException when messages is null.
|
|
/// </summary>
|
|
[Fact]
|
|
public async Task RunAsync_WithNullMessages_ThrowsArgumentNullExceptionAsync()
|
|
{
|
|
// Arrange
|
|
var mockAgent = new Mock<AIAgent>();
|
|
IEnumerable<OpenAIChatMessage>? messages = null;
|
|
|
|
// Act & Assert
|
|
var exception = await Assert.ThrowsAsync<ArgumentNullException>(
|
|
() => mockAgent.Object.RunAsync(messages!));
|
|
|
|
Assert.Equal("messages", exception.ParamName);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Verify that the RunAsync extension method calls the underlying agent's RunAsync with converted messages and parameters.
|
|
/// </summary>
|
|
[Fact]
|
|
public async Task RunAsync_CallsUnderlyingAgentAsync()
|
|
{
|
|
// Arrange
|
|
var mockAgent = new Mock<AIAgent>();
|
|
var mockSession = new Mock<AgentSession>();
|
|
var options = new AgentRunOptions();
|
|
var cancellationToken = new CancellationToken(false);
|
|
const string TestMessageText = "Hello, assistant!";
|
|
const string ResponseText = "This is the assistant's response.";
|
|
var openAiMessages = new List<OpenAIChatMessage>
|
|
{
|
|
OpenAIChatMessage.CreateUserMessage(TestMessageText)
|
|
};
|
|
|
|
var responseMessage = new ChatMessage(ChatRole.Assistant, [new TextContent(ResponseText)]);
|
|
|
|
mockAgent
|
|
.Protected()
|
|
.Setup<Task<AgentResponse>>("RunCoreAsync",
|
|
ItExpr.IsAny<IEnumerable<ChatMessage>>(),
|
|
ItExpr.IsAny<AgentSession?>(),
|
|
ItExpr.IsAny<AgentRunOptions?>(),
|
|
ItExpr.IsAny<CancellationToken>())
|
|
.ReturnsAsync(new AgentResponse([responseMessage]));
|
|
|
|
// Act
|
|
var result = await mockAgent.Object.RunAsync(openAiMessages, mockSession.Object, options, cancellationToken);
|
|
|
|
// Assert
|
|
mockAgent.Protected()
|
|
.Verify("RunCoreAsync",
|
|
Times.Once(),
|
|
ItExpr.Is<IEnumerable<ChatMessage>>(msgs =>
|
|
msgs.ToList().Count == 1 &&
|
|
msgs.ToList()[0].Text == TestMessageText),
|
|
mockSession.Object,
|
|
options,
|
|
cancellationToken
|
|
);
|
|
|
|
Assert.NotNull(result);
|
|
Assert.NotEmpty(result.Content);
|
|
Assert.Equal(ResponseText, result.Content.Last().Text);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Verify that RunStreamingAsync throws ArgumentNullException when agent is null.
|
|
/// </summary>
|
|
[Fact]
|
|
public void RunStreamingAsync_WithNullAgent_ThrowsArgumentNullException()
|
|
{
|
|
// Arrange
|
|
AIAgent? agent = null;
|
|
var messages = new List<OpenAIChatMessage>
|
|
{
|
|
OpenAIChatMessage.CreateUserMessage("Test message")
|
|
};
|
|
|
|
// Act & Assert
|
|
Assert.Throws<ArgumentNullException>(
|
|
"agent",
|
|
() => agent!.RunStreamingAsync(messages));
|
|
}
|
|
|
|
/// <summary>
|
|
/// Verify that RunStreamingAsync throws ArgumentNullException when messages is null.
|
|
/// </summary>
|
|
[Fact]
|
|
public void RunStreamingAsync_WithNullMessages_ThrowsArgumentNullException()
|
|
{
|
|
// Arrange
|
|
var mockAgent = new Mock<AIAgent>();
|
|
IEnumerable<OpenAIChatMessage>? messages = null;
|
|
|
|
// Act & Assert
|
|
var exception = Assert.Throws<ArgumentNullException>(
|
|
() => mockAgent.Object.RunStreamingAsync(messages!));
|
|
|
|
Assert.Equal("messages", exception.ParamName);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Verify that the RunStreamingAsync extension method calls the underlying agent's RunStreamingAsync with converted messages and parameters.
|
|
/// </summary>
|
|
[Fact]
|
|
public async Task RunStreamingAsync_CallsUnderlyingAgentAsync()
|
|
{
|
|
// Arrange
|
|
var mockAgent = new Mock<AIAgent>();
|
|
var mockSession = new Mock<AgentSession>();
|
|
var options = new AgentRunOptions();
|
|
var cancellationToken = new CancellationToken(false);
|
|
const string TestMessageText = "Hello, assistant!";
|
|
const string ResponseText1 = "This is ";
|
|
const string ResponseText2 = "the assistant's response.";
|
|
var openAiMessages = new List<OpenAIChatMessage>
|
|
{
|
|
OpenAIChatMessage.CreateUserMessage(TestMessageText)
|
|
};
|
|
|
|
var responseUpdates = new List<AgentResponseUpdate>
|
|
{
|
|
new(ChatRole.Assistant, ResponseText1),
|
|
new(ChatRole.Assistant, ResponseText2)
|
|
};
|
|
|
|
mockAgent
|
|
.Protected()
|
|
.Setup<IAsyncEnumerable<AgentResponseUpdate>>("RunCoreStreamingAsync",
|
|
ItExpr.IsAny<IEnumerable<ChatMessage>>(),
|
|
ItExpr.IsAny<AgentSession?>(),
|
|
ItExpr.IsAny<AgentRunOptions?>(),
|
|
ItExpr.IsAny<CancellationToken>())
|
|
.Returns(ToAsyncEnumerableAsync(responseUpdates));
|
|
|
|
// Act
|
|
var result = mockAgent.Object.RunStreamingAsync(openAiMessages, mockSession.Object, options, cancellationToken);
|
|
var updateCount = 0;
|
|
await foreach (var update in result)
|
|
{
|
|
updateCount++;
|
|
}
|
|
|
|
// Assert
|
|
mockAgent.Protected()
|
|
.Verify("RunCoreStreamingAsync",
|
|
Times.Once(),
|
|
ItExpr.Is<IEnumerable<ChatMessage>>(msgs =>
|
|
msgs.ToList().Count == 1 &&
|
|
msgs.ToList()[0].Text == TestMessageText),
|
|
mockSession.Object,
|
|
options,
|
|
cancellationToken
|
|
);
|
|
|
|
Assert.True(updateCount > 0, "Expected at least one streaming update");
|
|
}
|
|
|
|
/// <summary>
|
|
/// Helper method to convert a list of AgentResponseUpdate to an async enumerable.
|
|
/// </summary>
|
|
private static async IAsyncEnumerable<AgentResponseUpdate> ToAsyncEnumerableAsync(IEnumerable<AgentResponseUpdate> updates)
|
|
{
|
|
foreach (var update in updates)
|
|
{
|
|
yield return await Task.FromResult(update);
|
|
}
|
|
}
|
|
|
|
#region ResponseItem overload tests
|
|
|
|
/// <summary>
|
|
/// Verify that RunAsync with ResponseItem throws ArgumentNullException when agent is null.
|
|
/// </summary>
|
|
[Fact]
|
|
public async Task RunAsync_ResponseItem_WithNullAgent_ThrowsArgumentNullExceptionAsync()
|
|
{
|
|
// Arrange
|
|
AIAgent? agent = null;
|
|
IEnumerable<ResponseItem> messages = [ResponseItem.CreateUserMessageItem("Test message")];
|
|
|
|
// Act & Assert
|
|
var exception = await Assert.ThrowsAsync<ArgumentNullException>(
|
|
() => agent!.RunAsync(messages));
|
|
|
|
Assert.Equal("agent", exception.ParamName);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Verify that RunAsync with ResponseItem throws ArgumentNullException when messages is null.
|
|
/// </summary>
|
|
[Fact]
|
|
public async Task RunAsync_ResponseItem_WithNullMessages_ThrowsArgumentNullExceptionAsync()
|
|
{
|
|
// Arrange
|
|
var mockAgent = new Mock<AIAgent>();
|
|
IEnumerable<ResponseItem>? messages = null;
|
|
|
|
// Act & Assert
|
|
var exception = await Assert.ThrowsAsync<ArgumentNullException>(
|
|
() => mockAgent.Object.RunAsync(messages!));
|
|
|
|
Assert.Equal("messages", exception.ParamName);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Verify that the RunAsync with ResponseItem extension method calls the underlying agent's RunAsync with converted messages and parameters.
|
|
/// </summary>
|
|
[Fact]
|
|
public async Task RunAsync_ResponseItem_CallsUnderlyingAgentAsync()
|
|
{
|
|
// Arrange
|
|
var mockAgent = new Mock<AIAgent>();
|
|
var mockSession = new Mock<AgentSession>();
|
|
var options = new AgentRunOptions();
|
|
var cancellationToken = new CancellationToken(false);
|
|
const string TestMessageText = "Hello, assistant!";
|
|
const string ResponseText = "This is the assistant's response.";
|
|
IEnumerable<ResponseItem> responseItemMessages = [ResponseItem.CreateUserMessageItem(TestMessageText)];
|
|
|
|
var responseMessage = new ChatMessage(ChatRole.Assistant, [new TextContent(ResponseText)]);
|
|
|
|
mockAgent
|
|
.Protected()
|
|
.Setup<Task<AgentResponse>>("RunCoreAsync",
|
|
ItExpr.IsAny<IEnumerable<ChatMessage>>(),
|
|
ItExpr.IsAny<AgentSession?>(),
|
|
ItExpr.IsAny<AgentRunOptions?>(),
|
|
ItExpr.IsAny<CancellationToken>())
|
|
.ReturnsAsync(new AgentResponse([responseMessage]));
|
|
|
|
// Act
|
|
ResponseResult result = await mockAgent.Object.RunAsync(responseItemMessages, mockSession.Object, options, cancellationToken);
|
|
|
|
// Assert
|
|
mockAgent.Protected()
|
|
.Verify("RunCoreAsync",
|
|
Times.Once(),
|
|
ItExpr.IsAny<IEnumerable<ChatMessage>>(),
|
|
mockSession.Object,
|
|
options,
|
|
cancellationToken
|
|
);
|
|
|
|
Assert.NotNull(result);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Verify that RunStreamingAsync with ResponseItem throws ArgumentNullException when agent is null.
|
|
/// </summary>
|
|
[Fact]
|
|
public void RunStreamingAsync_ResponseItem_WithNullAgent_ThrowsArgumentNullException()
|
|
{
|
|
// Arrange
|
|
AIAgent? agent = null;
|
|
IEnumerable<ResponseItem> messages = [ResponseItem.CreateUserMessageItem("Test message")];
|
|
|
|
// Act & Assert
|
|
Assert.Throws<ArgumentNullException>(
|
|
"agent",
|
|
() => agent!.RunStreamingAsync(messages));
|
|
}
|
|
|
|
/// <summary>
|
|
/// Verify that RunStreamingAsync with ResponseItem throws ArgumentNullException when messages is null.
|
|
/// </summary>
|
|
[Fact]
|
|
public void RunStreamingAsync_ResponseItem_WithNullMessages_ThrowsArgumentNullException()
|
|
{
|
|
// Arrange
|
|
var mockAgent = new Mock<AIAgent>();
|
|
IEnumerable<ResponseItem>? messages = null;
|
|
|
|
// Act & Assert
|
|
var exception = Assert.Throws<ArgumentNullException>(
|
|
() => mockAgent.Object.RunStreamingAsync(messages!));
|
|
|
|
Assert.Equal("messages", exception.ParamName);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Verify that the RunStreamingAsync with ResponseItem extension method calls the underlying agent's RunStreamingAsync with converted messages and parameters.
|
|
/// </summary>
|
|
[Fact]
|
|
public async Task RunStreamingAsync_ResponseItem_CallsUnderlyingAgentAsync()
|
|
{
|
|
// Arrange
|
|
var mockAgent = new Mock<AIAgent>();
|
|
var mockSession = new Mock<AgentSession>();
|
|
var options = new AgentRunOptions();
|
|
var cancellationToken = new CancellationToken(false);
|
|
const string TestMessageText = "Hello, assistant!";
|
|
const string ResponseText1 = "This is ";
|
|
const string ResponseText2 = "the assistant's response.";
|
|
IEnumerable<ResponseItem> responseItemMessages = [ResponseItem.CreateUserMessageItem(TestMessageText)];
|
|
|
|
var responseUpdates = new List<AgentResponseUpdate>
|
|
{
|
|
new(ChatRole.Assistant, ResponseText1),
|
|
new(ChatRole.Assistant, ResponseText2)
|
|
};
|
|
|
|
mockAgent
|
|
.Protected()
|
|
.Setup<IAsyncEnumerable<AgentResponseUpdate>>("RunCoreStreamingAsync",
|
|
ItExpr.IsAny<IEnumerable<ChatMessage>>(),
|
|
ItExpr.IsAny<AgentSession?>(),
|
|
ItExpr.IsAny<AgentRunOptions?>(),
|
|
ItExpr.IsAny<CancellationToken>())
|
|
.Returns(ToAsyncEnumerableAsync(responseUpdates));
|
|
|
|
// Act
|
|
var result = mockAgent.Object.RunStreamingAsync(responseItemMessages, mockSession.Object, options, cancellationToken);
|
|
var updateCount = 0;
|
|
await foreach (var update in result)
|
|
{
|
|
updateCount++;
|
|
}
|
|
|
|
// Assert
|
|
mockAgent.Protected()
|
|
.Verify("RunCoreStreamingAsync",
|
|
Times.Once(),
|
|
ItExpr.IsAny<IEnumerable<ChatMessage>>(),
|
|
mockSession.Object,
|
|
options,
|
|
cancellationToken
|
|
);
|
|
}
|
|
|
|
#endregion
|
|
}
|