mirror of
https://github.com/microsoft/agent-framework.git
synced 2026-06-16 21:04:09 +08:00
7a8b456294
* WIP * Wip * Updated ADR * Updated ADR * Update files * Address copilot comments * Update filters from Task<T> to Task only * Project endpoint * Add agent ctor filter * Other Agent Framework investigation * Remove SK Java, no support * Update LlamaIndex info * Removing unrelated files * Implementation with specialization * Remove the specialization option as extra unecessary complexity * Move middleware responsibility to a decorator * Update readme * Function invocation wip * Add Agent Builder * Adding comparison samples * Reorganize Samples and Processor vs Decorator * Remove merge files * Address formating warnigs * Update ADR * Step13 README's update * Address PR feedback * Address PR feedback * Remove configure await from ADR samples * Update variables * Address feedback * Address Agent level tool invocation with Options.ToolsTransformer strategy * Removing the Processor approach * Proposal design for Middleware in CreateAIAgent extensions * Examples clean up and consolitation * Update middlewares to work with ApprovalREquiredFunction * Clean-up sample * Update override function call sample * Drop configuration from the extensions, looks overkill * Builder interface .. * Revert IAIBuilder interface approach * Cleanup sample * Adding unit tests * Fix UT * Cleanup sample * Remove unneeded dependency * Address PR comment + Readme Samples * Add missing comments for Program.cs Middleware * Address mor PR comments + add client factory for OpenAI extensions * Add OpenAI UnitTests for extensions * Add AzureAI PersistentChatClient UT * Addess feedback * Add function invoking UT * Add builder extension UT * Address feedback + Rearange abstractions + UT fixes * Drop context based middleware for full decorating impl * Update unit tests * Update UT coverage * Removing Middelware namespace * Add missing UT * Remove internal ToolTransformation Property * Adjust xmldoc * Remove transient file * Address merge conflict * Add xmldoc remark for clarity * Address comment * Address feedback * Update UT --------- Co-authored-by: Chris <66376200+crickman@users.noreply.github.com>
228 lines
7.6 KiB
C#
228 lines
7.6 KiB
C#
// Copyright (c) Microsoft. All rights reserved.
|
|
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.Runtime.CompilerServices;
|
|
using System.Threading;
|
|
using System.Threading.Tasks;
|
|
using Microsoft.Extensions.AI;
|
|
using OpenAI;
|
|
using OpenAIChatClient = OpenAI.Chat.ChatClient;
|
|
|
|
namespace Microsoft.Agents.AI.OpenAI.UnitTests.Extensions;
|
|
|
|
/// <summary>
|
|
/// Unit tests for the <see cref="OpenAIChatClientExtensions"/> class.
|
|
/// </summary>
|
|
public sealed class OpenAIChatClientExtensionsTests
|
|
{
|
|
/// <summary>
|
|
/// Test custom chat client that can be used to verify clientFactory functionality.
|
|
/// </summary>
|
|
private sealed class TestChatClient : IChatClient
|
|
{
|
|
private readonly IChatClient _innerClient;
|
|
|
|
public TestChatClient(IChatClient innerClient)
|
|
{
|
|
this._innerClient = innerClient;
|
|
}
|
|
|
|
public Task<ChatResponse> GetResponseAsync(IEnumerable<ChatMessage> messages, ChatOptions? options = null, CancellationToken cancellationToken = default)
|
|
=> this._innerClient.GetResponseAsync(messages, options, cancellationToken);
|
|
|
|
public async IAsyncEnumerable<ChatResponseUpdate> GetStreamingResponseAsync(
|
|
IEnumerable<ChatMessage> messages, ChatOptions? options = null, [EnumeratorCancellation] CancellationToken cancellationToken = default)
|
|
{
|
|
await foreach (var update in this._innerClient.GetStreamingResponseAsync(messages, options, cancellationToken))
|
|
{
|
|
yield return update;
|
|
}
|
|
}
|
|
|
|
public object? GetService(Type serviceType, object? serviceKey = null)
|
|
{
|
|
// Return this instance when requested
|
|
if (serviceType == typeof(TestChatClient))
|
|
{
|
|
return this;
|
|
}
|
|
|
|
return this._innerClient.GetService(serviceType, serviceKey);
|
|
}
|
|
|
|
public void Dispose() => this._innerClient.Dispose();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Creates a test ChatClient implementation for testing.
|
|
/// </summary>
|
|
private sealed class TestOpenAIChatClient : OpenAIChatClient
|
|
{
|
|
public TestOpenAIChatClient()
|
|
{
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Verify that CreateAIAgent with clientFactory parameter correctly applies the factory.
|
|
/// </summary>
|
|
[Fact]
|
|
public void CreateAIAgent_WithClientFactory_AppliesFactoryCorrectly()
|
|
{
|
|
// Arrange
|
|
var chatClient = new TestOpenAIChatClient();
|
|
var testChatClient = new TestChatClient(chatClient.AsIChatClient());
|
|
|
|
// Act
|
|
var agent = chatClient.CreateAIAgent(
|
|
instructions: "Test instructions",
|
|
name: "Test Agent",
|
|
description: "Test description",
|
|
clientFactory: (innerClient) => testChatClient);
|
|
|
|
// Assert
|
|
Assert.NotNull(agent);
|
|
Assert.Equal("Test Agent", agent.Name);
|
|
Assert.Equal("Test description", agent.Description);
|
|
|
|
// Verify that the custom chat client can be retrieved from the agent's service collection
|
|
var retrievedTestClient = agent.GetService<TestChatClient>();
|
|
Assert.NotNull(retrievedTestClient);
|
|
Assert.Same(testChatClient, retrievedTestClient);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Verify that CreateAIAgent with clientFactory using AsBuilder pattern works correctly.
|
|
/// </summary>
|
|
[Fact]
|
|
public void CreateAIAgent_WithClientFactoryUsingAsBuilder_AppliesFactoryCorrectly()
|
|
{
|
|
// Arrange
|
|
var chatClient = new TestOpenAIChatClient();
|
|
TestChatClient? testChatClient = null;
|
|
|
|
// Act
|
|
var agent = chatClient.CreateAIAgent(
|
|
instructions: "Test instructions",
|
|
clientFactory: (innerClient) =>
|
|
innerClient.AsBuilder().Use((innerClient) => testChatClient = new TestChatClient(innerClient)).Build());
|
|
|
|
// Assert
|
|
Assert.NotNull(agent);
|
|
|
|
// Verify that the custom chat client can be retrieved from the agent's service collection
|
|
var retrievedTestClient = agent.GetService<TestChatClient>();
|
|
Assert.NotNull(retrievedTestClient);
|
|
Assert.Same(testChatClient, retrievedTestClient);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Verify that CreateAIAgent with options and clientFactory parameter correctly applies the factory.
|
|
/// </summary>
|
|
[Fact]
|
|
public void CreateAIAgent_WithOptionsAndClientFactory_AppliesFactoryCorrectly()
|
|
{
|
|
// Arrange
|
|
var chatClient = new TestOpenAIChatClient();
|
|
var testChatClient = new TestChatClient(chatClient.AsIChatClient());
|
|
var options = new ChatClientAgentOptions
|
|
{
|
|
Name = "Test Agent",
|
|
Description = "Test description",
|
|
Instructions = "Test instructions"
|
|
};
|
|
|
|
// Act
|
|
var agent = chatClient.CreateAIAgent(
|
|
options,
|
|
clientFactory: (innerClient) => testChatClient);
|
|
|
|
// Assert
|
|
Assert.NotNull(agent);
|
|
Assert.Equal("Test Agent", agent.Name);
|
|
Assert.Equal("Test description", agent.Description);
|
|
|
|
// Verify that the custom chat client can be retrieved from the agent's service collection
|
|
var retrievedTestClient = agent.GetService<TestChatClient>();
|
|
Assert.NotNull(retrievedTestClient);
|
|
Assert.Same(testChatClient, retrievedTestClient);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Verify that CreateAIAgent without clientFactory works normally.
|
|
/// </summary>
|
|
[Fact]
|
|
public void CreateAIAgent_WithoutClientFactory_WorksNormally()
|
|
{
|
|
// Arrange
|
|
var chatClient = new TestOpenAIChatClient();
|
|
|
|
// Act
|
|
var agent = chatClient.CreateAIAgent(
|
|
instructions: "Test instructions",
|
|
name: "Test Agent");
|
|
|
|
// Assert
|
|
Assert.NotNull(agent);
|
|
Assert.Equal("Test Agent", agent.Name);
|
|
|
|
// Verify that no TestChatClient is available since no factory was provided
|
|
var retrievedTestClient = agent.GetService<TestChatClient>();
|
|
Assert.Null(retrievedTestClient);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Verify that CreateAIAgent with null clientFactory works normally.
|
|
/// </summary>
|
|
[Fact]
|
|
public void CreateAIAgent_WithNullClientFactory_WorksNormally()
|
|
{
|
|
// Arrange
|
|
var chatClient = new TestOpenAIChatClient();
|
|
|
|
// Act
|
|
var agent = chatClient.CreateAIAgent(
|
|
instructions: "Test instructions",
|
|
name: "Test Agent",
|
|
clientFactory: null);
|
|
|
|
// Assert
|
|
Assert.NotNull(agent);
|
|
Assert.Equal("Test Agent", agent.Name);
|
|
|
|
// Verify that no TestChatClient is available since no factory was provided
|
|
var retrievedTestClient = agent.GetService<TestChatClient>();
|
|
Assert.Null(retrievedTestClient);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Verify that CreateAIAgent throws ArgumentNullException when client is null.
|
|
/// </summary>
|
|
[Fact]
|
|
public void CreateAIAgent_WithNullClient_ThrowsArgumentNullException()
|
|
{
|
|
// Act & Assert
|
|
var exception = Assert.Throws<ArgumentNullException>(() =>
|
|
((OpenAIChatClient)null!).CreateAIAgent());
|
|
|
|
Assert.Equal("client", exception.ParamName);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Verify that CreateAIAgent with options throws ArgumentNullException when options is null.
|
|
/// </summary>
|
|
[Fact]
|
|
public void CreateAIAgent_WithNullOptions_ThrowsArgumentNullException()
|
|
{
|
|
// Arrange
|
|
var chatClient = new TestOpenAIChatClient();
|
|
|
|
// Act & Assert
|
|
var exception = Assert.Throws<ArgumentNullException>(() =>
|
|
chatClient.CreateAIAgent((ChatClientAgentOptions)null!));
|
|
|
|
Assert.Equal("options", exception.ParamName);
|
|
}
|
|
}
|