mirror of
https://github.com/microsoft/agent-framework.git
synced 2026-06-16 21:04:09 +08:00
1bf520a7c2
* add support for background responses * Update dotnet/src/Microsoft.Agents.AI.Abstractions/AgentRunResponseUpdate.cs Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> * fix broken link * fix xml comments and background responses properties override funcitonity * change ai model provider * use Run{Streaming}Async overloads that don't require messages * stop using m: prefix in cref attribute of <see/> element. * reject input messages provided with continuation token + don't extract messages from message store and context provide if continuation token is provided * use agent thread for background-responses sample * require agent thread for background responses * Update dotnet/src/Microsoft.Agents.AI/ChatClient/ChatClientAgent.cs Co-authored-by: Roger Barreto <19890735+rogerbarreto@users.noreply.github.com> * Update dotnet/src/Microsoft.Agents.AI/ChatClient/ChatClientAgent.cs Co-authored-by: Roger Barreto <19890735+rogerbarreto@users.noreply.github.com> * remove CA1200 * Update dotnet/src/Microsoft.Agents.AI.Abstractions/AgentRunOptions.cs Co-authored-by: westey <164392973+westey-m@users.noreply.github.com> * Update dotnet/src/Microsoft.Agents.AI.Abstractions/AgentRunResponse.cs Co-authored-by: westey <164392973+westey-m@users.noreply.github.com> * Update dotnet/src/Microsoft.Agents.AI.Abstractions/AgentRunResponse.cs Co-authored-by: westey <164392973+westey-m@users.noreply.github.com> * address pr review comments * Update dotnet/samples/GettingStarted/Agents/Agent_Step17_BackgroundResponses/Program.cs Co-authored-by: westey <164392973+westey-m@users.noreply.github.com> --------- Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> Co-authored-by: Roger Barreto <19890735+rogerbarreto@users.noreply.github.com> Co-authored-by: westey <164392973+westey-m@users.noreply.github.com>
302 lines
11 KiB
C#
302 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 AgentRunResponseTests
|
|
{
|
|
[Fact]
|
|
public void ConstructorWithNullEmptyArgsIsValid()
|
|
{
|
|
AgentRunResponse 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 AgentRunResponse((ChatMessage)null!));
|
|
}
|
|
|
|
[Fact]
|
|
public void ConstructorWithMessagesRoundtrips()
|
|
{
|
|
AgentRunResponse 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),
|
|
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 }),
|
|
};
|
|
|
|
AgentRunResponse response = new(chatResponse);
|
|
Assert.Same(chatResponse.AdditionalProperties, response.AdditionalProperties);
|
|
Assert.Equal(chatResponse.CreatedAt, response.CreatedAt);
|
|
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()
|
|
{
|
|
AgentRunResponse 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);
|
|
}
|
|
|
|
[Fact]
|
|
public void JsonSerializationRoundtrips()
|
|
{
|
|
AgentRunResponse 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);
|
|
|
|
AgentRunResponse? result = JsonSerializer.Deserialize<AgentRunResponse>(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()
|
|
{
|
|
AgentRunResponse 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()
|
|
{
|
|
AgentRunResponse 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()
|
|
{
|
|
AgentRunResponse response = new();
|
|
|
|
Assert.Equal(string.Empty, response.Text);
|
|
}
|
|
|
|
[Fact]
|
|
public void ToAgentRunResponseUpdatesProducesUpdates()
|
|
{
|
|
AgentRunResponse 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 },
|
|
Usage = new UsageDetails
|
|
{
|
|
TotalTokenCount = 100
|
|
},
|
|
};
|
|
|
|
AgentRunResponseUpdate[] updates = response.ToAgentRunResponseUpdates();
|
|
Assert.NotNull(updates);
|
|
Assert.Equal(2, updates.Length);
|
|
|
|
AgentRunResponseUpdate 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);
|
|
|
|
AgentRunResponseUpdate 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 ParseAsStructuredOutputSuccess()
|
|
{
|
|
// Arrange.
|
|
var expectedResult = new Animal { Id = 1, FullName = "Tigger", Species = Species.Tiger };
|
|
var response = new AgentRunResponse(new ChatMessage(ChatRole.Assistant, JsonSerializer.Serialize(expectedResult, TestJsonSerializerContext.Default.Animal)));
|
|
|
|
// Act.
|
|
var animal = response.Deserialize<Animal>(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 ParseAsStructuredOutputFailsWithEmptyString()
|
|
{
|
|
// Arrange.
|
|
var response = new AgentRunResponse(new ChatMessage(ChatRole.Assistant, string.Empty));
|
|
|
|
// Act & Assert.
|
|
var exception = Assert.Throws<InvalidOperationException>(() => response.Deserialize<Animal>(TestJsonSerializerContext.Default.Options));
|
|
Assert.Equal("The response did not contain JSON to be deserialized.", exception.Message);
|
|
}
|
|
|
|
[Fact]
|
|
public void ParseAsStructuredOutputFailsWithInvalidJson()
|
|
{
|
|
// Arrange.
|
|
var response = new AgentRunResponse(new ChatMessage(ChatRole.Assistant, "invalid json"));
|
|
|
|
// Act & Assert.
|
|
Assert.Throws<JsonException>(() => response.Deserialize<Animal>(TestJsonSerializerContext.Default.Options));
|
|
}
|
|
|
|
[Fact]
|
|
public void ParseAsStructuredOutputFailsWithIncorrectTypedJson()
|
|
{
|
|
// Arrange.
|
|
var response = new AgentRunResponse(new ChatMessage(ChatRole.Assistant, "[]"));
|
|
|
|
// Act & Assert.
|
|
Assert.Throws<JsonException>(() => response.Deserialize<Animal>(TestJsonSerializerContext.Default.Options));
|
|
}
|
|
|
|
[Fact]
|
|
public void TryParseAsStructuredOutputSuccess()
|
|
{
|
|
// Arrange.
|
|
var expectedResult = new Animal { Id = 1, FullName = "Tigger", Species = Species.Tiger };
|
|
var response = new AgentRunResponse(new ChatMessage(ChatRole.Assistant, JsonSerializer.Serialize(expectedResult, TestJsonSerializerContext.Default.Animal)));
|
|
|
|
// Act.
|
|
response.TryDeserialize(TestJsonSerializerContext.Default.Options, out Animal? animal);
|
|
|
|
// 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 TryParseAsStructuredOutputFailsWithEmptyText()
|
|
{
|
|
// Arrange.
|
|
var response = new AgentRunResponse(new ChatMessage(ChatRole.Assistant, string.Empty));
|
|
|
|
// Act & Assert.
|
|
Assert.False(response.TryDeserialize<Animal>(TestJsonSerializerContext.Default.Options, out _));
|
|
}
|
|
|
|
[Fact]
|
|
public void TryParseAsStructuredOutputFailsWithIncorrectTypedJson()
|
|
{
|
|
// Arrange.
|
|
var response = new AgentRunResponse(new ChatMessage(ChatRole.Assistant, "[]"));
|
|
|
|
// Act & Assert.
|
|
Assert.False(response.TryDeserialize<Animal>(TestJsonSerializerContext.Default.Options, out _));
|
|
}
|
|
}
|