usingMicrosoft.SemanticKernel;usingMicrosoft.SemanticKernel.ChatCompletion;usingMicrosoft.SemanticKernel.Connectors.OpenAI;usingstatic System.Environment;namespaceCustomCopilot{internalclassProgram{staticasync Task Main(string[] args){// Create a kernel with the Azure OpenAI chat completion servicevar builder = Kernel.CreateBuilder(); builder.AddAzureOpenAIChatCompletion("gpt-35-turbo-16k", GetEnvironmentVariable("AOI_ENDPOINT_SWDN")!, GetEnvironmentVariable("AOI_KEY_SWDN")!);// Build the kernelvar kernel = builder.Build();// Create chat history ChatHistory history =[]; history.AddSystemMessage(@"You're a virtual assistant that helps people find information.");// Get chat completion servicevar chatCompletionService = kernel.GetRequiredService<IChatCompletionService>();// Start the conversationwhile(true){// Get user input Console.ForegroundColor = ConsoleColor.White; Console.Write("User > "); history.AddUserMessage(Console.ReadLine()!);// Enable auto function calling OpenAIPromptExecutionSettings openAIPromptExecutionSettings =new(){ MaxTokens =200};// Get the response from the AIvar response = chatCompletionService.GetStreamingChatMessageContentsAsync( history, executionSettings: openAIPromptExecutionSettings, kernel: kernel); Console.ForegroundColor = ConsoleColor.Green; Console.Write("\nAssistant > ");string combinedResponse =string.Empty;awaitforeach(var message in response){//Write the response to the console Console.Write(message); combinedResponse += message;} Console.WriteLine();// Add the message from the agent to the chat history history.AddAssistantMessage(combinedResponse);}}}}
现在如果你只是在控制台应用程序中用 Program.cs 运行代码并提供必要的配置,它应该像 ChatGPT 一样工作。到目前为止,你可能看到在互联网上相同的演示,向你展示如何使用 Azure OpenAI API 并增强业务能力。让我们为它添加一些变化,继续问它现在几点了。
Semantic Kernel 包含一些必要概念和功能,例如 AI 插件、原生函数、提示等,这里不再详细介绍了,可以去官网查阅文档。只是简要解释一下,插件是一种为 AI Copilot(或代理)添加功能和能力的方法。它们现在基于 OpenAI 插件规范,这意味着它们可以与其他 AI 平台(如 ChatGPT、Microsoft 365 等)互操作。插件可以包含本机代码(例如 C# 或 Python)和提示(例如对 AI 服务的请求)。您可以使用插件来访问数据、执行操作或使用其他 AI 模型增强您的代理。例如可以创建一个插件搜索网络,发送电子邮件,生成图像甚至使用Microsoft Word,插件就像 AI 应用程序的手,允许它与现实世界进行交互。Semantic Kernel 插件的另一个优点是,其中一些插件已由团队作为单独的软件包发布。这意味着您不必自己编写所有内容,而是也可以使用一些内置的。例如在我们的例子中,我们将使用 TimePlugin 来解决这个问题。
你所要做的就是添加下面这些代码。
1
2
3
4
5
6
7
8
9
10
11
12
13
// Create a kernel with the Azure OpenAI chat completion servicevar builder = Kernel.CreateBuilder(); builder.AddAzureOpenAIChatCompletion("gpt-35-turbo-16k", GetEnvironmentVariable("AOI_ENDPOINT_SWDN")!, GetEnvironmentVariable("AOI_KEY_SWDN")!);// Load the plugins// add this namespace Microsoft.SemanticKernel.Plugins.Core;#pragma warning disable SKEXP0050
builder.Plugins.AddFromType<TimePlugin>();// Build the kernelvar kernel = builder.Build();
因此,我们将使用自动函数调用,要启用它,您只需要一行代码。在 while 循环中,只需设置一个 ToolCallBehavior 属性。
1
2
3
4
5
6
7
8
9
10
11
// Get user inputConsole.ForegroundColor = ConsoleColor.White;Console.Write("User > ");history.AddUserMessage(Console.ReadLine()!);// Enable auto function callingOpenAIPromptExecutionSettings openAIPromptExecutionSettings =new(){ MaxTokens =200, ToolCallBehavior = ToolCallBehavior.AutoInvokeKernelFunctions
};
我们现在将为每个功能创建一个插件。要创建插件,您需要定义一组可以向 AI 应用和服务公开的函数(或单个函数)。每个函数都应该有一个语义描述,解释它的作用、输入和输出以及它可能产生的副作用,即函数内部发生了什么及其返回值。这有助于 AI 了解如何使用您的插件并计划实现用户目标的最佳方式。您还可以为插件指定角色,该角色定义了插件的行为方式以及与用户的通信方式。例如,你可以让你的插件变得严苛或友好。
航班跟踪
该插件将使用 Aviation Stack API (免费计划)来获取航班的详细信息。对于免费版本,您需要提供 IATA 航班代码(例如 EK 414)或提供 IATA 源名称(例如 DXB/LHE)和 IATA 目的地名称。最初,我认为我们必须编写2个函数,一个是获取 IATA 名称 ,例如 API 所期望的悉尼的 SYD。否则,您将无法获得正确的响应。只需查看下面的屏幕截图,您可以看到我提供的 dep_iata 为迪拜, arr_iata 为悉尼。它没有返回任何东西。但是,当我将其更改为实际代码时,它返回了正确的响应。
很明显,除非您提供正确的 IATA 代码名称,否则您无法真正从 API 获得任何响应,但有趣的是,事实并非如此。借助函数语义描述的强大功能,它需要 LLM 和我们编写的代码的强大功能才能为您带来最佳结果。以下是我为航班跟踪写的代码:
usingMicrosoft.SemanticKernel;usingSystem;usingSystem.Collections.Generic;usingSystem.ComponentModel;usingSystem.Linq;usingSystem.Text;usingSystem.Threading.Tasks;namespaceCustomCopilot.Plugins.FlightTrackerPlugin{publicclassFlightTrackerPlugin(string apiKey){readonly HttpClient client =new HttpClient(); [KernelFunction, Description("Tracks the flight status of a provided source and destination")] [return: Description("Flight details and status")]publicasync Task<string> TrackFlightAsync( [Description("IATA code for the source location")]string source, [Description("IATA code for the designation location")]string destination, [Description("IATA code for the flight")]string flightNumber, [Description("Count of flights")]int limit){string url =$"""http://api.aviationstack.com/v1/flights
?access_key={apiKey}&dep_iata={source}&arr_iata={destination}
&limit={limit}&flight_iata={flightNumber}"""; HttpResponseMessage response =await client.GetAsync(url); response.EnsureSuccessStatusCode();string responseBody =await response.Content.ReadAsStringAsync();return responseBody;}}}
这只是一个类的一个函数。这意味着,一个插件只有一个本机功能。如果你注意到,从定义到它的参数再到它的返回值,我已经为这个函数中的几乎所有内容提供了描述 。这些语义描述的原因是 Semantic Kernel 将有效地使用 AI 来处理请求。这就是为什么,如果您注意到源和目的地,我会分别为源/目的地位置编写 IATA 代码。通过提供此描述,无需调用单独的服务来解析城市名称,在我们的示例悉尼中,它可以自动将其转换为 SYD。
You're a virtual assistant responsible for only flight tracking,
weather updates and finding out the right places within Australia
after inquiring about the proximity or city. You should not talk
anything outside of your scope. Your response should be very concise
and to the point. For each correct answer, you will get some $10
from me as a reward. Be nice with people.