From 702c010d77dd3b2c544c66048087f5f6b9b3f54b Mon Sep 17 00:00:00 2001 From: Xiaoyun Zhang Date: Thu, 23 May 2024 21:14:29 -0700 Subject: [PATCH] [.Net] add ollama-sample and adds more tests (#2776) * add ollama-sample and adds more tests * Update AutoGen.Ollama.Sample.csproj --- dotnet/AutoGen.sln | 6 +++ .../AutoGen.Ollama.Sample.csproj | 24 +++++++++ .../AutoGen.Ollama.Sample/Chat_With_LLaMA.cs | 28 ++++++++++ .../AutoGen.Ollama.Sample/Chat_With_LLaVA.cs | 40 +++++++++++++++ .../sample/AutoGen.Ollama.Sample/Program.cs | 6 +++ .../images/background.png | 3 ++ .../Middlewares/OllamaMessageConnector.cs | 35 +++++++------ .../OllamaMessageTests.cs | 51 ++++++++++++++----- 8 files changed, 163 insertions(+), 30 deletions(-) create mode 100644 dotnet/sample/AutoGen.Ollama.Sample/AutoGen.Ollama.Sample.csproj create mode 100644 dotnet/sample/AutoGen.Ollama.Sample/Chat_With_LLaMA.cs create mode 100644 dotnet/sample/AutoGen.Ollama.Sample/Chat_With_LLaVA.cs create mode 100644 dotnet/sample/AutoGen.Ollama.Sample/Program.cs create mode 100644 dotnet/sample/AutoGen.Ollama.Sample/images/background.png diff --git a/dotnet/AutoGen.sln b/dotnet/AutoGen.sln index 2f1413f4344f..be40e7b61b6d 100644 --- a/dotnet/AutoGen.sln +++ b/dotnet/AutoGen.sln @@ -43,6 +43,7 @@ Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "AutoGen.Ollama", "src\AutoG EndProject Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "AutoGen.Ollama.Tests", "test\AutoGen.Ollama.Tests\AutoGen.Ollama.Tests.csproj", "{03E31CAA-3728-48D3-B936-9F11CF6C18FE}" EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "AutoGen.Ollama.Sample", "sample\AutoGen.Ollama.Sample\AutoGen.Ollama.Sample.csproj", "{93AA4D0D-6EE4-44D5-AD77-7F73A3934544}" Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "AutoGen.SemanticKernel.Sample", "sample\AutoGen.SemanticKernel.Sample\AutoGen.SemanticKernel.Sample.csproj", "{52958A60-3FF7-4243-9058-34A6E4F55C31}" EndProject Global @@ -119,6 +120,10 @@ Global {03E31CAA-3728-48D3-B936-9F11CF6C18FE}.Debug|Any CPU.Build.0 = Debug|Any CPU {03E31CAA-3728-48D3-B936-9F11CF6C18FE}.Release|Any CPU.ActiveCfg = Release|Any CPU {03E31CAA-3728-48D3-B936-9F11CF6C18FE}.Release|Any CPU.Build.0 = Release|Any CPU + {93AA4D0D-6EE4-44D5-AD77-7F73A3934544}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {93AA4D0D-6EE4-44D5-AD77-7F73A3934544}.Debug|Any CPU.Build.0 = Debug|Any CPU + {93AA4D0D-6EE4-44D5-AD77-7F73A3934544}.Release|Any CPU.ActiveCfg = Release|Any CPU + {93AA4D0D-6EE4-44D5-AD77-7F73A3934544}.Release|Any CPU.Build.0 = Release|Any CPU {52958A60-3FF7-4243-9058-34A6E4F55C31}.Debug|Any CPU.ActiveCfg = Debug|Any CPU {52958A60-3FF7-4243-9058-34A6E4F55C31}.Debug|Any CPU.Build.0 = Debug|Any CPU {52958A60-3FF7-4243-9058-34A6E4F55C31}.Release|Any CPU.ActiveCfg = Release|Any CPU @@ -145,6 +150,7 @@ Global {B61388CA-DC73-4B7F-A7B2-7B9A86C9229E} = {F823671B-3ECA-4AE6-86DA-25E920D3FE64} {9F9E6DED-3D92-4970-909A-70FC11F1A665} = {18BF8DD7-0585-48BF-8F97-AD333080CE06} {03E31CAA-3728-48D3-B936-9F11CF6C18FE} = {F823671B-3ECA-4AE6-86DA-25E920D3FE64} + {93AA4D0D-6EE4-44D5-AD77-7F73A3934544} = {FBFEAD1F-29EB-4D99-A672-0CD8473E10B9} {52958A60-3FF7-4243-9058-34A6E4F55C31} = {FBFEAD1F-29EB-4D99-A672-0CD8473E10B9} EndGlobalSection GlobalSection(ExtensibilityGlobals) = postSolution diff --git a/dotnet/sample/AutoGen.Ollama.Sample/AutoGen.Ollama.Sample.csproj b/dotnet/sample/AutoGen.Ollama.Sample/AutoGen.Ollama.Sample.csproj new file mode 100644 index 000000000000..1dc94400869e --- /dev/null +++ b/dotnet/sample/AutoGen.Ollama.Sample/AutoGen.Ollama.Sample.csproj @@ -0,0 +1,24 @@ + + + Exe + $(TestTargetFramework) + enable + True + $(NoWarn);CS8981;CS8600;CS8602;CS8604;CS8618;CS0219;SKEXP0054;SKEXP0050;SKEXP0110 + + + + + + + + + + + + + PreserveNewest + + + + diff --git a/dotnet/sample/AutoGen.Ollama.Sample/Chat_With_LLaMA.cs b/dotnet/sample/AutoGen.Ollama.Sample/Chat_With_LLaMA.cs new file mode 100644 index 000000000000..e1af08c574c5 --- /dev/null +++ b/dotnet/sample/AutoGen.Ollama.Sample/Chat_With_LLaMA.cs @@ -0,0 +1,28 @@ +// Copyright (c) Microsoft Corporation. All rights reserved. +// Chat_With_LLaMA.cs + +using AutoGen.Core; +using AutoGen.Ollama.Extension; + +namespace AutoGen.Ollama.Sample; + +public class Chat_With_LLaMA +{ + public static async Task RunAsync() + { + using var httpClient = new HttpClient() + { + BaseAddress = new Uri("https://2xbvtxd1-11434.usw2.devtunnels.ms") + }; + + var ollamaAgent = new OllamaAgent( + httpClient: httpClient, + name: "ollama", + modelName: "llama3:latest", + systemMessage: "You are a helpful AI assistant") + .RegisterMessageConnector() + .RegisterPrintMessage(); + + var reply = await ollamaAgent.SendAsync("Can you write a piece of C# code to calculate 100th of fibonacci?"); + } +} diff --git a/dotnet/sample/AutoGen.Ollama.Sample/Chat_With_LLaVA.cs b/dotnet/sample/AutoGen.Ollama.Sample/Chat_With_LLaVA.cs new file mode 100644 index 000000000000..b1b310e3956d --- /dev/null +++ b/dotnet/sample/AutoGen.Ollama.Sample/Chat_With_LLaVA.cs @@ -0,0 +1,40 @@ +// Copyright (c) Microsoft Corporation. All rights reserved. +// Chat_With_LLaVA.cs + +using AutoGen.Core; +using AutoGen.Ollama.Extension; + +namespace AutoGen.Ollama.Sample; + +public class Chat_With_LLaVA +{ + public static async Task RunAsync() + { + using var httpClient = new HttpClient() + { + BaseAddress = new Uri("https://2xbvtxd1-11434.usw2.devtunnels.ms") + }; + + var ollamaAgent = new OllamaAgent( + httpClient: httpClient, + name: "ollama", + modelName: "llava:latest", + systemMessage: "You are a helpful AI assistant") + .RegisterMessageConnector() + .RegisterPrintMessage(); + + var image = Path.Combine("images", "background.png"); + var binaryData = BinaryData.FromBytes(File.ReadAllBytes(image), "image/png"); + var imageMessage = new ImageMessage(Role.User, binaryData); + var textMessage = new TextMessage(Role.User, "what's in this image?"); + var reply = await ollamaAgent.SendAsync(chatHistory: [textMessage, imageMessage]); + + // You can also use MultiModalMessage to put text and image together in one message + // In this case, all the messages in the multi-modal message will be put into single piece of message + // where the text is the concatenation of all the text messages seperated by \n + // and the images are all the images in the multi-modal message + var multiModalMessage = new MultiModalMessage(Role.User, [textMessage, imageMessage]); + + reply = await ollamaAgent.SendAsync(chatHistory: [multiModalMessage]); + } +} diff --git a/dotnet/sample/AutoGen.Ollama.Sample/Program.cs b/dotnet/sample/AutoGen.Ollama.Sample/Program.cs new file mode 100644 index 000000000000..62c92eebe7e5 --- /dev/null +++ b/dotnet/sample/AutoGen.Ollama.Sample/Program.cs @@ -0,0 +1,6 @@ +// Copyright (c) Microsoft Corporation. All rights reserved. +// Program.cs + +using AutoGen.Ollama.Sample; + +await Chat_With_LLaVA.RunAsync(); diff --git a/dotnet/sample/AutoGen.Ollama.Sample/images/background.png b/dotnet/sample/AutoGen.Ollama.Sample/images/background.png new file mode 100644 index 000000000000..ca276f81f5b0 --- /dev/null +++ b/dotnet/sample/AutoGen.Ollama.Sample/images/background.png @@ -0,0 +1,3 @@ +version https://git-lfs.github.com/spec/v1 +oid sha256:300b7c9d6ba0c23a3e52fbd2e268141ddcca0434a9fb9dcf7e58e7e903d36dcf +size 2126185 diff --git a/dotnet/src/AutoGen.Ollama/Middlewares/OllamaMessageConnector.cs b/dotnet/src/AutoGen.Ollama/Middlewares/OllamaMessageConnector.cs index e4e1c4ba47bb..a21ec3a1c991 100644 --- a/dotnet/src/AutoGen.Ollama/Middlewares/OllamaMessageConnector.cs +++ b/dotnet/src/AutoGen.Ollama/Middlewares/OllamaMessageConnector.cs @@ -64,13 +64,9 @@ public async IAsyncEnumerable InvokeAsync(MiddlewareContext c // if the chunks are not empty, aggregate them into a single message var messageContent = string.Join(string.Empty, chunks.Select(c => c.Message?.Value)); - var message = new Message - { - Role = "assistant", - Value = messageContent, - }; + var message = new TextMessage(Role.Assistant, messageContent, agent.Name); - yield return MessageEnvelope.Create(message, agent.Name); + yield return message; } private IEnumerable ProcessMessage(IEnumerable messages, IAgent agent) @@ -96,18 +92,25 @@ private IEnumerable ProcessMessage(IEnumerable messages, IAg private IEnumerable ProcessMultiModalMessage(MultiModalMessage multiModalMessage, IAgent agent) { - var messages = new List(); - foreach (var message in multiModalMessage.Content) + var textMessages = multiModalMessage.Content.Where(m => m is TextMessage textMessage && textMessage.GetContent() is not null); + var imageMessages = multiModalMessage.Content.Where(m => m is ImageMessage); + + // aggregate the text messages into one message + // by concatenating the content using newline + var textContent = string.Join("\n", textMessages.Select(m => ((TextMessage)m).Content)); + + // collect all the images + var images = imageMessages.SelectMany(m => ProcessImageMessage((ImageMessage)m, agent) + .SelectMany(m => (m as IMessage)?.Content.Images)); + + var message = new Message() { - messages.AddRange(message switch - { - TextMessage textMessage => ProcessTextMessage(textMessage, agent), - ImageMessage imageMessage => ProcessImageMessage(imageMessage, agent), - _ => throw new InvalidOperationException("Invalid message type"), - }); - } + Role = "user", + Value = textContent, + Images = images.ToList(), + }; - return messages; + return [MessageEnvelope.Create(message, agent.Name)]; } private IEnumerable ProcessImageMessage(ImageMessage imageMessage, IAgent agent) diff --git a/dotnet/test/AutoGen.Ollama.Tests/OllamaMessageTests.cs b/dotnet/test/AutoGen.Ollama.Tests/OllamaMessageTests.cs index 3f37db70275f..b19291e97671 100644 --- a/dotnet/test/AutoGen.Ollama.Tests/OllamaMessageTests.cs +++ b/dotnet/test/AutoGen.Ollama.Tests/OllamaMessageTests.cs @@ -2,13 +2,10 @@ // OllamaMessageTests.cs using AutoGen.Core; -using AutoGen.Ollama; using AutoGen.Tests; using FluentAssertions; using Xunit; -using Message = AutoGen.Ollama.Message; - -namespace Autogen.Ollama.Tests; +namespace AutoGen.Ollama.Tests; public class OllamaMessageTests { @@ -42,6 +39,36 @@ public async Task ItProcessUserTextMessageAsync() await agent.SendAsync(userMessage); } + [Fact] + public async Task ItProcessStreamingTextMessageAsync() + { + var messageConnector = new OllamaMessageConnector(); + var agent = new EchoAgent("assistant") + .RegisterStreamingMiddleware(messageConnector); + + var messageChunks = Enumerable.Range(0, 10) + .Select(i => new ChatResponseUpdate() + { + Message = new Message() + { + Value = i.ToString(), + Role = "assistant", + } + }) + .Select(m => MessageEnvelope.Create(m)); + + IStreamingMessage? finalReply = null; + await foreach (var reply in agent.GenerateStreamingReplyAsync(messageChunks)) + { + reply.Should().BeAssignableTo(); + finalReply = reply; + } + + finalReply.Should().BeOfType(); + var textMessage = (TextMessage)finalReply!; + textMessage.GetContent().Should().Be("0123456789"); + } + [Fact] public async Task ItProcessAssistantTextMessageAsync() { @@ -126,17 +153,13 @@ public async Task ItProcessMultiModalMessageAsync() var agent = new EchoAgent("assistant") .RegisterMiddleware(async (msgs, _, innerAgent, ct) => { - msgs.Count().Should().Be(2); - var textMessage = msgs.First(); - textMessage.Should().BeOfType>(); - var message = (IMessage)textMessage; - message.Content.Role.Should().Be("user"); + msgs.Count().Should().Be(1); + var message = msgs.First(); + message.Should().BeOfType>(); - var imageMessage = msgs.Last(); - imageMessage.Should().BeOfType>(); - message = (IMessage)imageMessage; - message.Content.Role.Should().Be("user"); - message.Content.Images!.Count.Should().Be(1); + var multiModalMessage = (IMessage)message; + multiModalMessage.Content.Images!.Count.Should().Be(1); + multiModalMessage.Content.Value.Should().Be("Hello"); return await innerAgent.GenerateReplyAsync(msgs); })