mirror of
https://github.com/microsoft/agent-framework.git
synced 2026-06-16 21:04:09 +08:00
77404d165c
* Notes * Readme typo * Update readme * Checkpoint * Namespace fix * Fix ID and namespace * Checkpoint * Verified * Comments * Isolate "Kit" * Address note: static * Checkpoint * Checkpoint "Executor<>" * Prefix and internal executors * Test passing * Cleanup * Rename "session" concept * Revert workflow debug * Fix template base / pragma * Tune system scope * Update dotnet/src/Microsoft.Agents.Workflows.Declarative/CodeGen/ResetVariableTemplate.tt Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> * Fix empty template * Add validation for codegen ut * Fix test * Codegen baselines * Constant * Prep * Mark TODO * Fix * Namespace * One more * Update baselines * Checkpoint * Checkpoint * Checkpoint * fme * Checkpoint * Another step * Fixed up * Roslyn * Fix * More cleaning * Async * Fix * Enum checkpoint * Refine enum * Checkpoint * Sync templates * Checkpoint * Streamline * Pre-merge analyzer updates * Foreach * Placeholders * Checkpoint * Clean-up * Sample path resolution * Checkpoint * Checkpoint - Workflow Code Building * Validation * Test cleanup * Update test basline * Update test baseline * Fix DefaultTemplate usage * Validation checkpoint * Fix break/continue edges * Verify generated code builds * Fix merge * Fix build validation * Update template handling of literal string values. * Test for metadata case * Update baselines * Fix merge * Checkpoint * Checkpoint: Conditions * Invoke Agent Checkpoint * Namespace * Address code-analysis issues * Cross platform test support * Invoke agent checkpoint * Clean sample * Checkpoint: Agent Invoke Input Messages * Checkpoint - Passing * Checkpoint * Regenerate all template + port conversation fix * Checkpoint: Tests good * Fix test for unbuntu * Fix build command * Checkpoint - E2E * Test fix * Update integration tests * Fix merge * Update * Checkpoint !!! * Baby steps * Checkpoint * Checkpoint E2E !!! * So close... * Integrate test validation * Fix merge * Rebase tests * Namespace * Namespace * Test cleanup * Sample comment cleanup * Checkpoint: List conversion * Include these * CheckPoint: ParseValue * Namespace * Fix sampel * More namspace * Comments * Test updates * Test fix * Better build * Shared code * Sort solution * Fix build * Prune solution * One more * Conversion matrix * Final table conversion --------- Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com>
252 lines
8.8 KiB
C#
252 lines
8.8 KiB
C#
// Copyright (c) Microsoft. All rights reserved.
|
|
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.Diagnostics;
|
|
using System.IO;
|
|
using System.Linq;
|
|
using System.Reflection;
|
|
using System.Threading.Tasks;
|
|
using Azure.AI.Agents.Persistent;
|
|
using Azure.Identity;
|
|
using Microsoft.Agents.AI.Workflows;
|
|
using Microsoft.Agents.AI.Workflows.Declarative;
|
|
using Microsoft.Extensions.AI;
|
|
using Microsoft.Extensions.Configuration;
|
|
using Test.WorkflowProviders;
|
|
|
|
namespace Demo.DeclarativeCode;
|
|
|
|
/// <summary>
|
|
/// HOW TO: Execute a declarative workflow that has been converted to code.
|
|
/// </summary>
|
|
/// <remarks>
|
|
/// <b>Configuration</b>
|
|
/// Define FOUNDRY_PROJECT_ENDPOINT as a user-secret or environment variable that
|
|
/// points to your Foundry project endpoint.
|
|
/// </remarks>
|
|
internal sealed class Program
|
|
{
|
|
public static async Task Main(string[] args)
|
|
{
|
|
Program program = new(args);
|
|
await program.ExecuteAsync();
|
|
}
|
|
|
|
private async Task ExecuteAsync()
|
|
{
|
|
// Use DeclarativeWorkflowBuilder to build a workflow based on a YAML file.
|
|
DeclarativeWorkflowOptions options =
|
|
new(new AzureAgentProvider(this.FoundryEndpoint, new AzureCliCredential()))
|
|
{
|
|
Configuration = this.Configuration
|
|
};
|
|
|
|
// Use the generated provider to create a workflow instance.
|
|
Workflow workflow = TestWorkflowProvider.CreateWorkflow<string>(options);
|
|
|
|
Notify("\nWORKFLOW: Starting...");
|
|
|
|
// Run the workflow, just like any other workflow
|
|
string input = this.GetWorkflowInput();
|
|
StreamingRun run = await InProcessExecution.StreamAsync(workflow, input);
|
|
await this.MonitorWorkflowRunAsync(run);
|
|
|
|
Notify("\nWORKFLOW: Done!");
|
|
}
|
|
|
|
private const string ConfigKeyFoundryEndpoint = "FOUNDRY_PROJECT_ENDPOINT";
|
|
|
|
private static readonly Dictionary<string, string> s_nameCache = [];
|
|
private static readonly HashSet<string> s_fileCache = [];
|
|
|
|
private string? WorkflowInput { get; }
|
|
private string FoundryEndpoint { get; }
|
|
private PersistentAgentsClient FoundryClient { get; }
|
|
private IConfiguration Configuration { get; }
|
|
|
|
private Program(string[] args)
|
|
{
|
|
this.WorkflowInput = ParseWorkflowInput(args);
|
|
|
|
this.Configuration = InitializeConfig();
|
|
|
|
this.FoundryEndpoint = this.Configuration[ConfigKeyFoundryEndpoint] ?? throw new InvalidOperationException($"Undefined configuration setting: {ConfigKeyFoundryEndpoint}");
|
|
this.FoundryClient = new PersistentAgentsClient(this.FoundryEndpoint, new AzureCliCredential());
|
|
}
|
|
|
|
private async Task MonitorWorkflowRunAsync(StreamingRun run)
|
|
{
|
|
string? messageId = null;
|
|
|
|
await foreach (WorkflowEvent evt in run.WatchStreamAsync().ConfigureAwait(false))
|
|
{
|
|
if (evt is ExecutorInvokedEvent executorInvoked)
|
|
{
|
|
Debug.WriteLine($"STEP ENTER #{executorInvoked.ExecutorId}");
|
|
}
|
|
else if (evt is ExecutorCompletedEvent executorComplete)
|
|
{
|
|
Debug.WriteLine($"STEP EXIT #{executorComplete.ExecutorId}");
|
|
}
|
|
else if (evt is ExecutorFailedEvent executorFailure)
|
|
{
|
|
Debug.WriteLine($"STEP ERROR #{executorFailure.ExecutorId}: {executorFailure.Data?.Message ?? "Unknown"}");
|
|
}
|
|
else if (evt is WorkflowErrorEvent workflowError)
|
|
{
|
|
Debug.WriteLine("WORKFLOW ERROR");
|
|
}
|
|
else if (evt is ConversationUpdateEvent invokeEvent)
|
|
{
|
|
Debug.WriteLine($"CONVERSATION: {invokeEvent.Data}");
|
|
}
|
|
else if (evt is AgentRunUpdateEvent streamEvent)
|
|
{
|
|
if (!string.Equals(messageId, streamEvent.Update.MessageId, StringComparison.Ordinal))
|
|
{
|
|
messageId = streamEvent.Update.MessageId;
|
|
|
|
if (messageId is not null)
|
|
{
|
|
string? agentId = streamEvent.Update.AuthorName;
|
|
if (agentId is not null)
|
|
{
|
|
if (!s_nameCache.TryGetValue(agentId, out string? realName))
|
|
{
|
|
PersistentAgent agent = await this.FoundryClient.Administration.GetAgentAsync(agentId);
|
|
s_nameCache[agentId] = agent.Name;
|
|
realName = agent.Name;
|
|
}
|
|
agentId = realName;
|
|
}
|
|
agentId ??= nameof(ChatRole.Assistant);
|
|
Console.ForegroundColor = ConsoleColor.Cyan;
|
|
Console.Write($"\n{agentId.ToUpperInvariant()}:");
|
|
Console.ForegroundColor = ConsoleColor.DarkGray;
|
|
Console.WriteLine($" [{messageId}]");
|
|
}
|
|
}
|
|
|
|
ChatResponseUpdate? chatUpdate = streamEvent.Update.RawRepresentation as ChatResponseUpdate;
|
|
switch (chatUpdate?.RawRepresentation)
|
|
{
|
|
case MessageContentUpdate messageUpdate:
|
|
string? fileId = messageUpdate.ImageFileId ?? messageUpdate.TextAnnotation?.OutputFileId;
|
|
if (fileId is not null && s_fileCache.Add(fileId))
|
|
{
|
|
BinaryData content = await this.FoundryClient.Files.GetFileContentAsync(fileId);
|
|
await DownloadFileContentAsync(Path.GetFileName(messageUpdate.TextAnnotation?.TextToReplace ?? "response.png"), content);
|
|
}
|
|
break;
|
|
}
|
|
try
|
|
{
|
|
Console.ResetColor();
|
|
Console.Write(streamEvent.Data);
|
|
}
|
|
finally
|
|
{
|
|
Console.ResetColor();
|
|
}
|
|
}
|
|
else if (evt is AgentRunResponseEvent messageEvent)
|
|
{
|
|
try
|
|
{
|
|
Console.WriteLine();
|
|
if (messageEvent.Response.AgentId is null)
|
|
{
|
|
Console.ForegroundColor = ConsoleColor.Cyan;
|
|
Console.WriteLine("ACTIVITY:");
|
|
Console.ForegroundColor = ConsoleColor.Yellow;
|
|
Console.WriteLine(messageEvent.Response?.Text.Trim());
|
|
}
|
|
else
|
|
{
|
|
if (messageEvent.Response.Usage is not null)
|
|
{
|
|
Console.ForegroundColor = ConsoleColor.DarkGray;
|
|
Console.WriteLine($"[Tokens Total: {messageEvent.Response.Usage.TotalTokenCount}, Input: {messageEvent.Response.Usage.InputTokenCount}, Output: {messageEvent.Response.Usage.OutputTokenCount}]");
|
|
}
|
|
}
|
|
}
|
|
finally
|
|
{
|
|
Console.ResetColor();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
private string GetWorkflowInput()
|
|
{
|
|
string? input = this.WorkflowInput;
|
|
|
|
try
|
|
{
|
|
Console.ForegroundColor = ConsoleColor.DarkGreen;
|
|
|
|
Console.Write("\nINPUT: ");
|
|
|
|
Console.ForegroundColor = ConsoleColor.White;
|
|
|
|
if (!string.IsNullOrWhiteSpace(input))
|
|
{
|
|
Console.WriteLine(input);
|
|
return input;
|
|
}
|
|
while (string.IsNullOrWhiteSpace(input))
|
|
{
|
|
input = Console.ReadLine();
|
|
}
|
|
|
|
return input.Trim();
|
|
}
|
|
finally
|
|
{
|
|
Console.ResetColor();
|
|
}
|
|
}
|
|
|
|
private static string? ParseWorkflowInput(string[] args)
|
|
{
|
|
return args?.FirstOrDefault();
|
|
}
|
|
|
|
// Load configuration from user-secrets
|
|
private static IConfigurationRoot InitializeConfig() =>
|
|
new ConfigurationBuilder()
|
|
.AddUserSecrets(Assembly.GetExecutingAssembly())
|
|
.AddEnvironmentVariables()
|
|
.Build();
|
|
|
|
private static void Notify(string message)
|
|
{
|
|
Console.ForegroundColor = ConsoleColor.Cyan;
|
|
try
|
|
{
|
|
Console.WriteLine(message);
|
|
}
|
|
finally
|
|
{
|
|
Console.ResetColor();
|
|
}
|
|
}
|
|
|
|
private static async ValueTask DownloadFileContentAsync(string filename, BinaryData content)
|
|
{
|
|
string filePath = Path.Combine(Path.GetTempPath(), Path.GetFileName(filename));
|
|
filePath = Path.ChangeExtension(filePath, ".png");
|
|
|
|
await File.WriteAllBytesAsync(filePath, content.ToArray());
|
|
|
|
Process.Start(
|
|
new ProcessStartInfo
|
|
{
|
|
FileName = "cmd.exe",
|
|
Arguments = $"/C start {filePath}"
|
|
});
|
|
}
|
|
}
|