mirror of
https://github.com/microsoft/agent-framework.git
synced 2026-06-16 21:04:09 +08:00
2f8fd5f82f
* Add FinishReason to AgentResponses * Address PR comments
291 lines
11 KiB
C#
291 lines
11 KiB
C#
// Copyright (c) Microsoft. All rights reserved.
|
|
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using System.Text.Json;
|
|
using Microsoft.Agents.AI.Abstractions.UnitTests.Models;
|
|
using Microsoft.Extensions.AI;
|
|
|
|
namespace Microsoft.Agents.AI.Abstractions.UnitTests;
|
|
|
|
public class AgentResponseTests
|
|
{
|
|
[Fact]
|
|
public void ConstructorWithNullEmptyArgsIsValid()
|
|
{
|
|
AgentResponse response;
|
|
|
|
response = new();
|
|
Assert.Empty(response.Messages);
|
|
Assert.Empty(response.Text);
|
|
Assert.Null(response.ContinuationToken);
|
|
|
|
response = new((IList<ChatMessage>?)null);
|
|
Assert.Empty(response.Messages);
|
|
Assert.Empty(response.Text);
|
|
Assert.Null(response.ContinuationToken);
|
|
|
|
Assert.Throws<ArgumentNullException>("message", () => new AgentResponse((ChatMessage)null!));
|
|
}
|
|
|
|
[Fact]
|
|
public void ConstructorWithMessagesRoundtrips()
|
|
{
|
|
AgentResponse response = new();
|
|
Assert.NotNull(response.Messages);
|
|
Assert.Same(response.Messages, response.Messages);
|
|
|
|
List<ChatMessage> messages = [];
|
|
response = new(messages);
|
|
Assert.Same(messages, response.Messages);
|
|
|
|
messages = [];
|
|
Assert.NotSame(messages, response.Messages);
|
|
response.Messages = messages;
|
|
Assert.Same(messages, response.Messages);
|
|
}
|
|
|
|
[Fact]
|
|
public void ConstructorWithChatResponseRoundtrips()
|
|
{
|
|
ChatResponse chatResponse = new()
|
|
{
|
|
AdditionalProperties = [],
|
|
CreatedAt = new DateTimeOffset(2022, 1, 1, 0, 0, 0, TimeSpan.Zero),
|
|
FinishReason = ChatFinishReason.ContentFilter,
|
|
Messages = [new(ChatRole.Assistant, "This is a test message.")],
|
|
RawRepresentation = new object(),
|
|
ResponseId = "responseId",
|
|
Usage = new UsageDetails(),
|
|
ContinuationToken = ResponseContinuationToken.FromBytes(new byte[] { 1, 2, 3 })
|
|
};
|
|
|
|
AgentResponse response = new(chatResponse);
|
|
Assert.Same(chatResponse.AdditionalProperties, response.AdditionalProperties);
|
|
Assert.Equal(chatResponse.CreatedAt, response.CreatedAt);
|
|
Assert.Equal(chatResponse.FinishReason, response.FinishReason);
|
|
Assert.Same(chatResponse.Messages, response.Messages);
|
|
Assert.Equal(chatResponse.ResponseId, response.ResponseId);
|
|
Assert.Same(chatResponse, response.RawRepresentation as ChatResponse);
|
|
Assert.Same(chatResponse.Usage, response.Usage);
|
|
Assert.Equivalent(ResponseContinuationToken.FromBytes(new byte[] { 1, 2, 3 }), response.ContinuationToken);
|
|
}
|
|
|
|
[Fact]
|
|
public void PropertiesRoundtrip()
|
|
{
|
|
AgentResponse response = new();
|
|
|
|
Assert.Null(response.AgentId);
|
|
response.AgentId = "agentId";
|
|
Assert.Equal("agentId", response.AgentId);
|
|
|
|
Assert.Null(response.ResponseId);
|
|
response.ResponseId = "id";
|
|
Assert.Equal("id", response.ResponseId);
|
|
|
|
Assert.Null(response.CreatedAt);
|
|
response.CreatedAt = new DateTimeOffset(2022, 1, 1, 0, 0, 0, TimeSpan.Zero);
|
|
Assert.Equal(new DateTimeOffset(2022, 1, 1, 0, 0, 0, TimeSpan.Zero), response.CreatedAt);
|
|
|
|
Assert.Null(response.Usage);
|
|
UsageDetails usage = new();
|
|
response.Usage = usage;
|
|
Assert.Same(usage, response.Usage);
|
|
|
|
Assert.Null(response.RawRepresentation);
|
|
object raw = new();
|
|
response.RawRepresentation = raw;
|
|
Assert.Same(raw, response.RawRepresentation);
|
|
|
|
Assert.Null(response.AdditionalProperties);
|
|
AdditionalPropertiesDictionary additionalProps = [];
|
|
response.AdditionalProperties = additionalProps;
|
|
Assert.Same(additionalProps, response.AdditionalProperties);
|
|
|
|
Assert.Null(response.ContinuationToken);
|
|
response.ContinuationToken = ResponseContinuationToken.FromBytes(new byte[] { 1, 2, 3 });
|
|
Assert.Equivalent(ResponseContinuationToken.FromBytes(new byte[] { 1, 2, 3 }), response.ContinuationToken);
|
|
|
|
Assert.Null(response.FinishReason);
|
|
response.FinishReason = ChatFinishReason.Length;
|
|
Assert.Equal(ChatFinishReason.Length, response.FinishReason);
|
|
}
|
|
|
|
[Fact]
|
|
public void JsonSerializationRoundtrips()
|
|
{
|
|
AgentResponse original = new(new ChatMessage(ChatRole.Assistant, "the message"))
|
|
{
|
|
AgentId = "agentId",
|
|
ResponseId = "id",
|
|
CreatedAt = new DateTimeOffset(2022, 1, 1, 0, 0, 0, TimeSpan.Zero),
|
|
Usage = new UsageDetails(),
|
|
RawRepresentation = new(),
|
|
AdditionalProperties = new() { ["key"] = "value" },
|
|
ContinuationToken = ResponseContinuationToken.FromBytes(new byte[] { 1, 2, 3 }),
|
|
};
|
|
|
|
string json = JsonSerializer.Serialize(original, AgentAbstractionsJsonUtilities.DefaultOptions);
|
|
|
|
AgentResponse? result = JsonSerializer.Deserialize<AgentResponse>(json, AgentAbstractionsJsonUtilities.DefaultOptions);
|
|
|
|
Assert.NotNull(result);
|
|
Assert.Equal(ChatRole.Assistant, result.Messages.Single().Role);
|
|
Assert.Equal("the message", result.Messages.Single().Text);
|
|
|
|
Assert.Equal("agentId", result.AgentId);
|
|
Assert.Equal("id", result.ResponseId);
|
|
Assert.Equal(new DateTimeOffset(2022, 1, 1, 0, 0, 0, TimeSpan.Zero), result.CreatedAt);
|
|
Assert.NotNull(result.Usage);
|
|
|
|
Assert.NotNull(result.AdditionalProperties);
|
|
Assert.Single(result.AdditionalProperties);
|
|
Assert.True(result.AdditionalProperties.TryGetValue("key", out object? value));
|
|
Assert.IsType<JsonElement>(value);
|
|
Assert.Equal("value", ((JsonElement)value!).GetString());
|
|
Assert.Equivalent(ResponseContinuationToken.FromBytes(new byte[] { 1, 2, 3 }), result.ContinuationToken);
|
|
}
|
|
|
|
[Fact]
|
|
public void ToStringOutputsText()
|
|
{
|
|
AgentResponse response = new(new ChatMessage(ChatRole.Assistant, $"This is a test.{Environment.NewLine}It's multiple lines."));
|
|
|
|
Assert.Equal(response.Text, response.ToString());
|
|
}
|
|
|
|
[Fact]
|
|
public void TextGetConcatenatesAllTextContent()
|
|
{
|
|
AgentResponse response = new(
|
|
[
|
|
new ChatMessage(
|
|
ChatRole.Assistant,
|
|
[
|
|
new DataContent("data:image/audio;base64,aGVsbG8="),
|
|
new DataContent("data:image/image;base64,aGVsbG8="),
|
|
new FunctionCallContent("callId1", "fc1"),
|
|
new TextContent("message1-text-1"),
|
|
new TextContent("message1-text-2"),
|
|
new FunctionResultContent("callId1", "result"),
|
|
]),
|
|
new ChatMessage(ChatRole.Assistant, "message2")
|
|
]);
|
|
|
|
Assert.Equal($"message1-text-1message1-text-2{Environment.NewLine}message2", response.Text);
|
|
}
|
|
|
|
[Fact]
|
|
public void TextGetReturnsEmptyStringWithNoMessages()
|
|
{
|
|
AgentResponse response = new();
|
|
|
|
Assert.Equal(string.Empty, response.Text);
|
|
}
|
|
|
|
[Fact]
|
|
public void ToAgentResponseUpdatesProducesUpdates()
|
|
{
|
|
AgentResponse response = new(new ChatMessage(new ChatRole("customRole"), "Text") { MessageId = "someMessage" })
|
|
{
|
|
AgentId = "agentId",
|
|
ResponseId = "12345",
|
|
CreatedAt = new DateTimeOffset(2024, 11, 10, 9, 20, 0, TimeSpan.Zero),
|
|
AdditionalProperties = new() { ["key1"] = "value1", ["key2"] = 42 },
|
|
FinishReason = ChatFinishReason.ContentFilter,
|
|
Usage = new UsageDetails
|
|
{
|
|
TotalTokenCount = 100
|
|
},
|
|
};
|
|
|
|
AgentResponseUpdate[] updates = response.ToAgentResponseUpdates();
|
|
Assert.NotNull(updates);
|
|
Assert.Equal(2, updates.Length);
|
|
|
|
AgentResponseUpdate update0 = updates[0];
|
|
Assert.Equal("agentId", update0.AgentId);
|
|
Assert.Equal("12345", update0.ResponseId);
|
|
Assert.Equal("someMessage", update0.MessageId);
|
|
Assert.Equal(new DateTimeOffset(2024, 11, 10, 9, 20, 0, TimeSpan.Zero), update0.CreatedAt);
|
|
Assert.Equal("customRole", update0.Role?.Value);
|
|
Assert.Equal("Text", update0.Text);
|
|
Assert.Equal(ChatFinishReason.ContentFilter, update0.FinishReason);
|
|
|
|
AgentResponseUpdate update1 = updates[1];
|
|
Assert.Equal("value1", update1.AdditionalProperties?["key1"]);
|
|
Assert.Equal(42, update1.AdditionalProperties?["key2"]);
|
|
Assert.IsType<UsageContent>(update1.Contents[0]);
|
|
UsageContent usageContent = (UsageContent)update1.Contents[0];
|
|
Assert.Equal(100, usageContent.Details.TotalTokenCount);
|
|
}
|
|
|
|
[Fact]
|
|
public void ParseAsStructuredOutputWithJSOSuccess()
|
|
{
|
|
// Arrange.
|
|
var expectedResult = new Animal { Id = 1, FullName = "Tigger", Species = Species.Tiger };
|
|
var response = new AgentResponse(new ChatMessage(ChatRole.Assistant, JsonSerializer.Serialize(expectedResult, TestJsonSerializerContext.Default.Animal)));
|
|
|
|
// Act.
|
|
var animal = JsonSerializer.Deserialize<Animal>(response.Text, TestJsonSerializerContext.Default.Options);
|
|
|
|
// Assert.
|
|
Assert.NotNull(animal);
|
|
Assert.Equal(expectedResult.Id, animal.Id);
|
|
Assert.Equal(expectedResult.FullName, animal.FullName);
|
|
Assert.Equal(expectedResult.Species, animal.Species);
|
|
}
|
|
|
|
[Fact]
|
|
public void ToAgentResponseUpdatesWithNoMessagesProducesEmptyArray()
|
|
{
|
|
// Arrange
|
|
AgentResponse response = new();
|
|
|
|
// Act
|
|
AgentResponseUpdate[] updates = response.ToAgentResponseUpdates();
|
|
|
|
// Assert
|
|
Assert.Empty(updates);
|
|
}
|
|
|
|
[Fact]
|
|
public void ToAgentResponseUpdatesWithUsageOnlyProducesSingleUpdate()
|
|
{
|
|
// Arrange
|
|
AgentResponse response = new()
|
|
{
|
|
Usage = new UsageDetails { TotalTokenCount = 100 }
|
|
};
|
|
|
|
// Act
|
|
AgentResponseUpdate[] updates = response.ToAgentResponseUpdates();
|
|
|
|
// Assert
|
|
AgentResponseUpdate update = Assert.Single(updates);
|
|
UsageContent usageContent = Assert.IsType<UsageContent>(update.Contents[0]);
|
|
Assert.Equal(100, usageContent.Details.TotalTokenCount);
|
|
}
|
|
|
|
[Fact]
|
|
public void ToAgentResponseUpdatesWithAdditionalPropertiesOnlyProducesSingleUpdate()
|
|
{
|
|
// Arrange
|
|
AgentResponse response = new()
|
|
{
|
|
AdditionalProperties = new() { ["key"] = "value" }
|
|
};
|
|
|
|
// Act
|
|
AgentResponseUpdate[] updates = response.ToAgentResponseUpdates();
|
|
|
|
// Assert
|
|
AgentResponseUpdate update = Assert.Single(updates);
|
|
Assert.NotNull(update.AdditionalProperties);
|
|
Assert.Equal("value", update.AdditionalProperties!["key"]);
|
|
}
|
|
}
|