mirror of
https://github.com/microsoft/agent-framework.git
synced 2026-06-16 21:04:09 +08:00
904a5b843e
* Python: .NET Samples - Restructure and Improve Samples (Feature Branch) (#4091) * Moved by agent (#4094) * Fix readme links * .NET Samples - Create `04-hosting` learning path step (#4098) * Agent move * Agent reorderd * Remove A2A section from README Removed A2A section from the Getting Started README. * Agent fixed links * Fix broken sample links in durable-agents README (#4101) * Initial plan * Fix broken internal links in documentation Co-authored-by: crickman <66376200+crickman@users.noreply.github.com> * Revert template link changes; keep only durable-agents README fix Co-authored-by: crickman <66376200+crickman@users.noreply.github.com> --------- Co-authored-by: copilot-swe-agent[bot] <198982749+Copilot@users.noreply.github.com> Co-authored-by: crickman <66376200+crickman@users.noreply.github.com> * .NET Samples - Create `03-workflows` learning path step (#4102) * Fix solution project path * Python: Fix broken markdown links to repo resources (outside /docs) (#4105) * Initial plan * Fix broken markdown links to repo resources Co-authored-by: crickman <66376200+crickman@users.noreply.github.com> * Update README to rename .NET Workflows Samples section --------- Co-authored-by: copilot-swe-agent[bot] <198982749+Copilot@users.noreply.github.com> Co-authored-by: crickman <66376200+crickman@users.noreply.github.com> * .NET Samples - Create `02-agents` learning path step (#4107) * .NET: Fix broken relative link in GroupChatToolApproval README (#4108) * Initial plan * Fix broken link in GroupChatToolApproval README Co-authored-by: crickman <66376200+crickman@users.noreply.github.com> --------- Co-authored-by: copilot-swe-agent[bot] <198982749+Copilot@users.noreply.github.com> Co-authored-by: crickman <66376200+crickman@users.noreply.github.com> * Update labeler configuration for workflow samples * .NET - Reorder Agents samples to start from Step01 instead of Step04 (#4110) * Fix solution * Resolve new sample paths * Move new AgentSkills and AgentWithMemory_Step04 samples * Fix link * Fix readme path * fix: update stale dotnet/samples/Durable path reference in AGENTS.md Co-authored-by: crickman <66376200+crickman@users.noreply.github.com> * Moved new sample * Update solution * Resolve merge (new sample) * Sync to new sample - FoundryAgents_Step21_BingCustomSearch * Updated README * .NET Samples - Configuration Naming Update (#4149) * .NET: Restore AzureFunctions index parity with ConsoleApps under DurableAgents samples (#4221) * Clean-up `05_host_your_agent` * Config setting consistency * Refine samples * AGENTS.md * Move new samples * Re-order samples * Move new project and fixup solution * Fixup model config * Fix up new UT project --------- Co-authored-by: Copilot <198982749+Copilot@users.noreply.github.com>
119 lines
4.7 KiB
C#
119 lines
4.7 KiB
C#
// Copyright (c) Microsoft. All rights reserved.
|
|
|
|
using Microsoft.Agents.AI.Workflows;
|
|
|
|
namespace WorkflowSharedStatesSample;
|
|
|
|
/// <summary>
|
|
/// This sample introduces the concept of shared states within a workflow.
|
|
/// It demonstrates how multiple executors can read from and write to shared states,
|
|
/// allowing for more complex data sharing and coordination between tasks.
|
|
/// </summary>
|
|
/// <remarks>
|
|
/// Pre-requisites:
|
|
/// - Foundational samples should be completed first.
|
|
/// - This sample also uses the fan-out and fan-in patterns to achieve parallel processing.
|
|
/// </remarks>
|
|
public static class Program
|
|
{
|
|
private static async Task Main()
|
|
{
|
|
// Create the executors
|
|
var fileRead = new FileReadExecutor();
|
|
var wordCount = new WordCountingExecutor();
|
|
var paragraphCount = new ParagraphCountingExecutor();
|
|
var aggregate = new AggregationExecutor();
|
|
|
|
// Build the workflow by connecting executors sequentially
|
|
var workflow = new WorkflowBuilder(fileRead)
|
|
.AddFanOutEdge(fileRead, [wordCount, paragraphCount])
|
|
.AddFanInBarrierEdge([wordCount, paragraphCount], aggregate)
|
|
.WithOutputFrom(aggregate)
|
|
.Build();
|
|
|
|
// Execute the workflow with input data
|
|
await using Run run = await InProcessExecution.RunAsync(workflow, "Lorem_Ipsum.txt");
|
|
foreach (WorkflowEvent evt in run.NewEvents)
|
|
{
|
|
if (evt is WorkflowOutputEvent outputEvent)
|
|
{
|
|
Console.WriteLine(outputEvent.Data);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Constants for shared state scopes.
|
|
/// </summary>
|
|
internal static class FileContentStateConstants
|
|
{
|
|
public const string FileContentStateScope = "FileContentState";
|
|
}
|
|
|
|
internal sealed class FileReadExecutor() : Executor<string, string>("FileReadExecutor")
|
|
{
|
|
public override async ValueTask<string> HandleAsync(string message, IWorkflowContext context, CancellationToken cancellationToken = default)
|
|
{
|
|
// Read file content from embedded resource
|
|
string fileContent = Resources.Read(message);
|
|
// Store file content in a shared state for access by other executors
|
|
string fileID = Guid.NewGuid().ToString("N");
|
|
await context.QueueStateUpdateAsync(fileID, fileContent, scopeName: FileContentStateConstants.FileContentStateScope, cancellationToken);
|
|
|
|
return fileID;
|
|
}
|
|
}
|
|
|
|
internal sealed class FileStats
|
|
{
|
|
public int ParagraphCount { get; set; }
|
|
public int WordCount { get; set; }
|
|
}
|
|
|
|
internal sealed class WordCountingExecutor() : Executor<string, FileStats>("WordCountingExecutor")
|
|
{
|
|
public override async ValueTask<FileStats> HandleAsync(string message, IWorkflowContext context, CancellationToken cancellationToken = default)
|
|
{
|
|
// Retrieve the file content from the shared state
|
|
var fileContent = await context.ReadStateAsync<string>(message, scopeName: FileContentStateConstants.FileContentStateScope, cancellationToken)
|
|
?? throw new InvalidOperationException("File content state not found");
|
|
|
|
int wordCount = fileContent.Split([' ', '\n', '\r'], StringSplitOptions.RemoveEmptyEntries).Length;
|
|
|
|
return new FileStats { WordCount = wordCount };
|
|
}
|
|
}
|
|
|
|
internal sealed class ParagraphCountingExecutor() : Executor<string, FileStats>("ParagraphCountingExecutor")
|
|
{
|
|
public override async ValueTask<FileStats> HandleAsync(string message, IWorkflowContext context, CancellationToken cancellationToken = default)
|
|
{
|
|
// Retrieve the file content from the shared state
|
|
var fileContent = await context.ReadStateAsync<string>(message, scopeName: FileContentStateConstants.FileContentStateScope, cancellationToken)
|
|
?? throw new InvalidOperationException("File content state not found");
|
|
|
|
int paragraphCount = fileContent.Split(['\n', '\r'], StringSplitOptions.RemoveEmptyEntries).Length;
|
|
|
|
return new FileStats { ParagraphCount = paragraphCount };
|
|
}
|
|
}
|
|
|
|
internal sealed class AggregationExecutor() : Executor<FileStats>("AggregationExecutor")
|
|
{
|
|
private readonly List<FileStats> _messages = [];
|
|
|
|
public override async ValueTask HandleAsync(FileStats message, IWorkflowContext context, CancellationToken cancellationToken = default)
|
|
{
|
|
this._messages.Add(message);
|
|
|
|
if (this._messages.Count == 2)
|
|
{
|
|
// Aggregate the results from both executors
|
|
var totalParagraphCount = this._messages.Sum(m => m.ParagraphCount);
|
|
var totalWordCount = this._messages.Sum(m => m.WordCount);
|
|
await context.YieldOutputAsync($"Total Paragraphs: {totalParagraphCount}, Total Words: {totalWordCount}", cancellationToken);
|
|
}
|
|
}
|
|
}
|