Files
agent-framework/dotnet/tests/Microsoft.Agents.AI.OpenAI.UnitTests/Extensions/AIAgentWithOpenAIExtensionsTests.cs
westey a3a9147e61 .NET: [BREAKING] Rename AgentThread to AgentSession (#3430)
* 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.
2026-01-26 16:30:25 +00:00

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
}