autogen/dotnet/test/AutoGen.Mistral.Tests/MistralClientTests.cs

287 lines
12 KiB
C#
Raw Normal View History

2024-09-30 16:32:48 -07:00
// Copyright (c) Microsoft Corporation. All rights reserved.
Bring Dotnet AutoGen (#924) * update readme * update * update * update * update * update * update * add sample project * revert notebook change back * update * update interactive version * add nuget package * refactor Message * update example * add azure nightly build pipeline * Set up CI with Azure Pipelines [skip ci] * Update nightly-build.yml for Azure Pipelines * add dotnet interactive package * add dotnet interactive package * update pipeline * add nuget feed back * remove dotnet-tool feed * remove dotnet-tool feed comment * update pipeline * update build name * Update nightly-build.yml * Delete .github/workflows/dotnet-ci.yml * update * add working_dir to use step * add initateChat api * update oai package * Update dotnet-build.yml * Update dotnet-run-openai-test-and-notebooks.yml * update build workflow * update build workflow * update nuget feed * update nuget feed * update aoai and sk version * Update InteractiveService.cs * add support for GPT 4V * add DalleAndGPT4V example * update example * add user proxy agent * add readme * bump version * update example * add dotnet interactive hook * update * udpate tests * add website * update index.md * add docs * update doc * move sk dependency out of core package * udpate doc * Update Use-function-call.md * add type safe function call document * update doc * update doc * add dock * Update Use-function-call.md * add GenerateReplyOptions * remove IChatLLM * update version * update doc * update website * add sample * fix link * add middleware agent * clean up doc * bump version * update doc * update * add Other Language * remove warnings * add sign.props * add sign step * fix pipelien * auth * real sign * disable PR trigger * update * disable PR trigger * use microbuild machine * update build pipeline to add publish to internal feed * add internal feed * fix build pipeline * add dotnet prefix * update ci * add build number * update run number * update source * update token * update * remove adding source * add publish to github package * try again * try again * ask for write pacakge * disable package when branch is not main * update * implement streaming agent * add test for streaming function call * update * fix #1588 * enable PR check for dotnet branch * add website readme * only publish to dotnet feed when pushing to dotnet branch * remove openai-test-and-notebooks workflow * update readme * update readme * update workflow * update getting-start * upgrade test and sample proejct to use .net 8 * fix global.json format && make loadFromConfig API internal only before implementing * update * add support for LM studio * add doc * Update README.md * add push and workflow_dispatch trigger * disable PR for main * add dotnet env * Update Installation.md * add nuget * refer to newtonsoft 13 * update branch to dotnet in docfx * Update Installation.md * pull out HumanInputMiddleware and FunctionCallMiddleware * fix tests * add link to sample folder * refactor message * refactor over IMessage * add more tests * add more test * fix build error * rename header * add semantic kernel project * update sk example * update dotnet version * add LMStudio function call example * rename LLaMAFunctin * remove dotnet run openai test and notebook workflow * add FunctionContract and test * update doc * add documents * add workflow * update * update sample * fix warning in test * reult length can be less then maximumOutputToKeep (#1804) * merge with main * add option to retrieve inner agent and middlewares from MiddlewareAgent * update doc * adjust namespace * update readme * fix test * use IMessage * more updates * update * fix test * add comments * use FunctionContract to replace FunctionDefinition * move AutoGen contrac to AutoGen.Core * update installation * refactor streamingAgent by adding StreamingMessage type * update sample * update samples * update * update * add test * fix test * bump version * add openaichat test * update * Update Example03_Agent_FunctionCall.cs * [.Net] improve docs (#1862) * add doc * add doc * add doc * add doc * add doc * add doc * update * fix test error * fix some error * fix test * fix test * add more tests * edits --------- Co-authored-by: ekzhu <ekzhu@users.noreply.github.com> * [.Net] Add fill form example (#1911) * add form filler example * update * fix ci error * [.Net] Add using AutoGen.Core in source generator (#1983) * fix using namespace bug in source generator * remove using in sourcegenerator test * disable PR test * Add .idea to .gitignore (#1988) * [.Net] publish to nuget.org feed (#1987) * publish to nuget * update ci * update dotnet-release * update release pipeline * add source * remove empty symbol package * update pipeline * remove tag * update installation guide * [.Net] Rename some classes && APIs based on doc review (#1980) * rename sequential group chat to round robin group chat * rename to sendInstruction * rename workflow to graph * rename some api * bump version * move Graph to GroupChat folder * rename fill application example * [.Net] Improve package description (#2161) * add discord link and update package description * Update getting-start.md * [.Net] Fix document comment from the most recent AutoGen.Net engineer sync (#2231) * update * rename RegisterPrintMessageHook to RegisterPrintMessage * update website * update update.md * fix link error * [.Net] Enable JsonMode and deterministic output in AutoGen.OpenAI OpenAIChatAgent (#2347) * update openai version && add sample for json output * add example in web * update update.md * update image url * [.Net] Add AutoGen.Mistral package (#2330) * add mstral client * enable streaming support * add mistralClientAgent * add test for function call * add extension * add support for toolcall and toolcall result message * add support for aggregate message * implement streaming function call * track (#2471) * [.Net] add mistral example (#2482) * update existing examples to use messageCOnnector * add overview * add function call document * add example 14 * add mistral token count usage example * update version * Update dotnet-release.yml (#2488) * update * revert gitattributes --------- Co-authored-by: mhensen <mh@webvize.nl> Co-authored-by: ekzhu <ekzhu@users.noreply.github.com> Co-authored-by: Krzysztof Kasprowicz <60486987+Krzysztof318@users.noreply.github.com>
2024-04-26 09:21:46 -07:00
// MistralClientTests.cs
using System.Text.Json;
using System.Text.Json.Serialization;
using AutoGen.Core;
using AutoGen.Mistral.Extension;
using AutoGen.Tests;
using FluentAssertions;
namespace AutoGen.Mistral.Tests;
public partial class MistralClientTests
{
[Function]
public async Task<string> GetWeather(string city)
{
return $"The weather in {city} is sunny.";
}
[ApiKeyFact("MISTRAL_API_KEY")]
public async Task MistralClientChatCompletionTestAsync()
{
var apiKey = Environment.GetEnvironmentVariable("MISTRAL_API_KEY") ?? throw new InvalidOperationException("MISTRAL_API_KEY is not set.");
var client = new MistralClient(apiKey: apiKey);
var systemMessage = new ChatMessage(ChatMessage.RoleEnum.System, "You are a helpful assistant.");
var userMessage = new ChatMessage(ChatMessage.RoleEnum.User, "What is the weather like today?");
var request = new ChatCompletionRequest(
model: "open-mistral-7b",
messages: new List<ChatMessage> { systemMessage, userMessage },
temperature: 0);
var response = await client.CreateChatCompletionsAsync(request);
response.Choices!.Count().Should().Be(1);
response.Choices!.First().Message!.Content.Should().NotBeNullOrEmpty();
response.Choices!.First().Message!.Role.Should().Be(ChatMessage.RoleEnum.Assistant);
response.Usage!.TotalTokens.Should().BeGreaterThan(0);
}
[ApiKeyFact("MISTRAL_API_KEY")]
public async Task MistralClientStreamingChatCompletionTestAsync()
{
var apiKey = Environment.GetEnvironmentVariable("MISTRAL_API_KEY") ?? throw new InvalidOperationException("MISTRAL_API_KEY is not set.");
var client = new MistralClient(apiKey: apiKey);
var systemMessage = new ChatMessage(ChatMessage.RoleEnum.System, "You are a helpful assistant.");
var userMessage = new ChatMessage(ChatMessage.RoleEnum.User, "What is the weather like today?");
var request = new ChatCompletionRequest(
model: "open-mistral-7b",
messages: new List<ChatMessage> { systemMessage, userMessage },
temperature: 0);
var response = client.StreamingChatCompletionsAsync(request);
var results = new List<ChatCompletionResponse>();
await foreach (var item in response)
{
results.Add(item);
item.VarObject.Should().Be("chat.completion.chunk");
}
results.Count.Should().BeGreaterThan(0);
// merge result
var finalResult = results.First();
foreach (var result in results)
{
if (finalResult.Choices!.First().Message is null)
{
finalResult.Choices!.First().Message = result.Choices!.First().Delta;
}
else
{
finalResult.Choices!.First().Message!.Content += result.Choices!.First().Delta!.Content;
}
// the usage information will be included in the last result
if (result.Usage != null)
{
finalResult.Usage = result.Usage;
}
}
finalResult.Choices!.First().Message!.Content.Should().NotBeNullOrEmpty();
finalResult.Choices!.First().Message!.Role.Should().Be(ChatMessage.RoleEnum.Assistant);
finalResult.Usage!.TotalTokens.Should().BeGreaterThan(0);
}
[ApiKeyFact("MISTRAL_API_KEY")]
public async Task MistralClientStreamingChatJsonModeCompletionTestAsync()
{
var apiKey = Environment.GetEnvironmentVariable("MISTRAL_API_KEY") ?? throw new InvalidOperationException("MISTRAL_API_KEY is not set.");
var client = new MistralClient(apiKey: apiKey);
var systemMessage = new ChatMessage(ChatMessage.RoleEnum.System, "You are a helpful assistant that convert input to json object");
var userMessage = new ChatMessage(ChatMessage.RoleEnum.User, "name: John, age: 41, email: g123456@gmail.com");
var request = new ChatCompletionRequest(
model: "open-mistral-7b",
messages: new List<ChatMessage> { systemMessage, userMessage },
temperature: 0)
{
ResponseFormat = new ResponseFormat { ResponseFormatType = "json_object" },
};
var response = client.StreamingChatCompletionsAsync(request);
var results = new List<ChatCompletionResponse>();
await foreach (var item in response)
{
results.Add(item);
item.VarObject.Should().Be("chat.completion.chunk");
}
results.Count.Should().BeGreaterThan(0);
// merge result
var finalResult = results.First();
foreach (var result in results)
{
if (finalResult.Choices!.First().Message is null)
{
finalResult.Choices!.First().Message = result.Choices!.First().Delta;
}
else
{
finalResult.Choices!.First().Message!.Content += result.Choices!.First().Delta!.Content;
}
// the usage information will be included in the last result
if (result.Usage != null)
{
finalResult.Usage = result.Usage;
}
}
finalResult.Choices!.First().Message!.Content.Should().NotBeNullOrEmpty();
finalResult.Choices!.First().Message!.Role.Should().Be(ChatMessage.RoleEnum.Assistant);
finalResult.Usage!.TotalTokens.Should().BeGreaterThan(0);
var responseContent = finalResult.Choices!.First().Message!.Content ?? throw new InvalidOperationException("Response content is null.");
var person = JsonSerializer.Deserialize<Person>(responseContent);
person.Should().NotBeNull();
person!.Name.Should().Be("John");
person!.Age.Should().Be(41);
person!.Email.Should().Be("g123456@gmail.com");
}
[ApiKeyFact("MISTRAL_API_KEY")]
public async Task MistralClientJsonModeTestAsync()
{
var apiKey = Environment.GetEnvironmentVariable("MISTRAL_API_KEY") ?? throw new InvalidOperationException("MISTRAL_API_KEY is not set.");
var client = new MistralClient(apiKey: apiKey);
var systemMessage = new ChatMessage(ChatMessage.RoleEnum.System, "You are a helpful assistant that convert input to json object");
var userMessage = new ChatMessage(ChatMessage.RoleEnum.User, "name: John, age: 41, email: g123456@gmail.com");
var request = new ChatCompletionRequest(
model: "open-mistral-7b",
messages: new List<ChatMessage> { systemMessage, userMessage },
temperature: 0)
{
ResponseFormat = new ResponseFormat { ResponseFormatType = "json_object" },
};
var response = await client.CreateChatCompletionsAsync(request);
response.Choices!.Count().Should().Be(1);
response.Choices!.First().Message!.Content.Should().NotBeNullOrEmpty();
response.Choices!.First().Message!.Role.Should().Be(ChatMessage.RoleEnum.Assistant);
response.Usage!.TotalTokens.Should().BeGreaterThan(0);
// check if the response is a valid json object
var responseContent = response.Choices!.First().Message!.Content ?? throw new InvalidOperationException("Response content is null.");
var person = JsonSerializer.Deserialize<Person>(responseContent);
person.Should().NotBeNull();
person!.Name.Should().Be("John");
person!.Age.Should().Be(41);
person!.Email.Should().Be("g123456@gmail.com");
}
[ApiKeyFact("MISTRAL_API_KEY")]
public async Task MistralClientFunctionCallTestAsync()
{
var apiKey = Environment.GetEnvironmentVariable("MISTRAL_API_KEY") ?? throw new InvalidOperationException("MISTRAL_API_KEY is not set.");
using var client = new MistralClient(apiKey: apiKey);
var getWeatherFunctionContract = this.GetWeatherFunctionContract;
var functionDefinition = getWeatherFunctionContract.ToMistralFunctionDefinition();
var systemMessage = new ChatMessage(ChatMessage.RoleEnum.System, "You are a helpful assistant.");
var userMessage = new ChatMessage(ChatMessage.RoleEnum.User, "What is the weather in Seattle?");
var request = new ChatCompletionRequest(
model: "mistral-small-latest", // only large or small latest models support function calls
messages: new List<ChatMessage> { systemMessage, userMessage },
temperature: 0)
{
Tools = [new FunctionTool(functionDefinition)],
ToolChoice = ToolChoiceEnum.Any,
};
var response = await client.CreateChatCompletionsAsync(request);
response.Choices!.Count().Should().Be(1);
response.Choices!.First().Message!.Content.Should().BeNullOrEmpty();
response.Choices!.First().FinishReason.Should().Be(Choice.FinishReasonEnum.ToolCalls);
response.Choices!.First().Message!.ToolCalls!.Count.Should().Be(1);
response.Choices!.First().Message!.ToolCalls!.First().Function.Name.Should().Be("GetWeather");
}
[ApiKeyFact("MISTRAL_API_KEY")]
public async Task MistralClientStreamingFunctionCallTestAsync()
{
var apiKey = Environment.GetEnvironmentVariable("MISTRAL_API_KEY") ?? throw new InvalidOperationException("MISTRAL_API_KEY is not set.");
using var client = new MistralClient(apiKey: apiKey);
var getWeatherFunctionContract = this.GetWeatherFunctionContract;
var functionDefinition = getWeatherFunctionContract.ToMistralFunctionDefinition();
var systemMessage = new ChatMessage(ChatMessage.RoleEnum.System, "You are a helpful assistant.");
var userMessage = new ChatMessage(ChatMessage.RoleEnum.User, "What is the weather in Seattle?");
var request = new ChatCompletionRequest(
model: "mistral-small-latest",
messages: new List<ChatMessage> { systemMessage, userMessage },
temperature: 0)
{
Tools = [new FunctionTool(functionDefinition)],
ToolChoice = ToolChoiceEnum.Any,
};
var response = client.StreamingChatCompletionsAsync(request);
var results = new List<ChatCompletionResponse>();
await foreach (var item in response)
{
results.Add(item);
item.VarObject.Should().Be("chat.completion.chunk");
}
// merge result
var finalResult = results.First();
var lastResult = results.Last();
lastResult.Choices!.First().FinishReason.Should().Be(Choice.FinishReasonEnum.ToolCalls);
foreach (var result in results)
{
if (finalResult.Choices!.First().Message is null)
{
finalResult.Choices!.First().Message = result.Choices!.First().Delta;
finalResult.Choices!.First().Message!.ToolCalls = [];
}
else
{
finalResult.Choices!.First().Message!.ToolCalls = finalResult.Choices!.First().Message!.ToolCalls!.Concat(result.Choices!.First().Delta!.ToolCalls!).ToList();
}
// the usage information will be included in the last result
if (result.Usage != null)
{
finalResult.Usage = result.Usage;
}
}
finalResult.Choices!.First().Message!.Content.Should().BeNullOrEmpty();
finalResult.Choices!.First().Message!.ToolCalls!.Count.Should().BeGreaterThan(0);
finalResult.Usage!.TotalTokens.Should().BeGreaterThan(0);
finalResult.Choices!.First().Message!.ToolCalls!.First().Function.Name.Should().Be("GetWeather");
}
}
public class Person
{
[JsonPropertyName("name")]
public string Name { get; set; } = string.Empty;
[JsonPropertyName("age")]
public int Age { get; set; }
[JsonPropertyName("email")]
public string Email { get; set; } = string.Empty;
}