-
Notifications
You must be signed in to change notification settings - Fork 1.1k
.NET: Durable Agent samples and automated validation for non-Azure Functions #3042
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Merged
cgillum
merged 20 commits into
microsoft:main
from
cgillum:cgillum/durable-agent-standalone
Jan 20, 2026
Merged
Changes from all commits
Commits
Show all changes
20 commits
Select commit
Hold shift + click to select a range
d851eac
Durable Agent samples and automated validation for non-Azure Functions
cgillum 9f6f345
Update test projects
cgillum a96cef8
fix file encoding
cgillum 053acf4
Merge branch 'main' into cgillum/durable-agent-standalone
cgillum e0034e5
Merge branch 'main' into cgillum/durable-agent-standalone
markwallace-microsoft 5b82a7e
Merge branch 'main' into cgillum/durable-agent-standalone
cgillum c0d6a60
Remove AgentThreadMetadata usage
cgillum ae1bb27
Merge branch 'main' into cgillum/durable-agent-standalone
cgillum 33fbab5
Merge branch 'main' into cgillum/durable-agent-standalone
cgillum 3f4c41e
Absorb breaking change from #3152
cgillum 5b3abfa
Merge branch 'main' into cgillum/durable-agent-standalone
cgillum bebb7db
Absorb newer breaking changes (AgentRunResponse --> AgentResponse)
cgillum 10e5004
Merge branch 'main' into cgillum/durable-agent-standalone
cgillum a858045
Merge branch 'main' into cgillum/durable-agent-standalone
cgillum 5675b8c
Absorb more breaking changes (see #3222)
cgillum 29b527d
Improve integration test reliability (isolated task hubs, etc.)
cgillum 3ea04a0
Merge branch 'main' into cgillum/durable-agent-standalone
cgillum 8d309e3
Fix flakey streaming test
cgillum d64a716
Merge branch 'cgillum/durable-agent-standalone' of https://github.com…
cgillum ecca920
Merge branch 'main' into cgillum/durable-agent-standalone
cgillum File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
There are no files selected for viewing
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
30 changes: 30 additions & 0 deletions
30
dotnet/samples/DurableAgents/ConsoleApps/01_SingleAgent/01_SingleAgent.csproj
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| Original file line number | Diff line number | Diff line change |
|---|---|---|
| @@ -0,0 +1,30 @@ | ||
| <Project Sdk="Microsoft.NET.Sdk"> | ||
| <PropertyGroup> | ||
| <TargetFrameworks>net10.0</TargetFrameworks> | ||
| <OutputType>Exe</OutputType> | ||
| <ImplicitUsings>enable</ImplicitUsings> | ||
| <Nullable>enable</Nullable> | ||
| <AssemblyName>SingleAgent</AssemblyName> | ||
| <RootNamespace>SingleAgent</RootNamespace> | ||
| </PropertyGroup> | ||
|
|
||
| <ItemGroup> | ||
| <PackageReference Include="Azure.AI.OpenAI" /> | ||
| <PackageReference Include="Azure.Identity" /> | ||
| <PackageReference Include="Microsoft.DurableTask.Client.AzureManaged" /> | ||
| <PackageReference Include="Microsoft.DurableTask.Worker.AzureManaged" /> | ||
| <PackageReference Include="Microsoft.Extensions.Hosting" /> | ||
| </ItemGroup> | ||
|
|
||
| <!-- Local projects that should be switched to package references when using the sample outside of this MAF repo --> | ||
| <!-- | ||
| <ItemGroup> | ||
| <PackageReference Include="Microsoft.Agents.AI.DurableTask" /> | ||
| <PackageReference Include="Microsoft.Agents.AI.OpenAI" /> | ||
| </ItemGroup> | ||
| --> | ||
| <ItemGroup> | ||
| <ProjectReference Include="..\..\..\..\src\Microsoft.Agents.AI.DurableTask\Microsoft.Agents.AI.DurableTask.csproj" /> | ||
| <ProjectReference Include="..\..\..\..\src\Microsoft.Agents.AI.OpenAI\Microsoft.Agents.AI.OpenAI.csproj" /> | ||
| </ItemGroup> | ||
| </Project> | ||
103 changes: 103 additions & 0 deletions
103
dotnet/samples/DurableAgents/ConsoleApps/01_SingleAgent/Program.cs
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| Original file line number | Diff line number | Diff line change |
|---|---|---|
| @@ -0,0 +1,103 @@ | ||
| // Copyright (c) Microsoft. All rights reserved. | ||
|
|
||
| using Azure; | ||
| using Azure.AI.OpenAI; | ||
| using Azure.Identity; | ||
| using Microsoft.Agents.AI; | ||
| using Microsoft.Agents.AI.DurableTask; | ||
| using Microsoft.DurableTask.Client.AzureManaged; | ||
| using Microsoft.DurableTask.Worker.AzureManaged; | ||
| using Microsoft.Extensions.AI; | ||
| using Microsoft.Extensions.DependencyInjection; | ||
| using Microsoft.Extensions.Hosting; | ||
| using Microsoft.Extensions.Logging; | ||
| using OpenAI.Chat; | ||
|
|
||
| // Get the Azure OpenAI endpoint and deployment name from environment variables. | ||
| string endpoint = Environment.GetEnvironmentVariable("AZURE_OPENAI_ENDPOINT") | ||
| ?? throw new InvalidOperationException("AZURE_OPENAI_ENDPOINT is not set."); | ||
| string deploymentName = Environment.GetEnvironmentVariable("AZURE_OPENAI_DEPLOYMENT") | ||
| ?? throw new InvalidOperationException("AZURE_OPENAI_DEPLOYMENT is not set."); | ||
|
|
||
| // Get DTS connection string from environment variable | ||
| string dtsConnectionString = Environment.GetEnvironmentVariable("DURABLE_TASK_SCHEDULER_CONNECTION_STRING") | ||
| ?? "Endpoint=http://localhost:8080;TaskHub=default;Authentication=None"; | ||
|
|
||
| // Use Azure Key Credential if provided, otherwise use Azure CLI Credential. | ||
| string? azureOpenAiKey = Environment.GetEnvironmentVariable("AZURE_OPENAI_KEY"); | ||
| AzureOpenAIClient client = !string.IsNullOrEmpty(azureOpenAiKey) | ||
| ? new AzureOpenAIClient(new Uri(endpoint), new AzureKeyCredential(azureOpenAiKey)) | ||
| : new AzureOpenAIClient(new Uri(endpoint), new AzureCliCredential()); | ||
|
|
||
| // Set up an AI agent following the standard Microsoft Agent Framework pattern. | ||
| const string JokerName = "Joker"; | ||
| const string JokerInstructions = "You are good at telling jokes."; | ||
|
|
||
| AIAgent agent = client.GetChatClient(deploymentName).AsAIAgent(JokerInstructions, JokerName); | ||
|
|
||
| // Configure the console app to host the AI agent. | ||
| IHost host = Host.CreateDefaultBuilder(args) | ||
| .ConfigureLogging(logging => logging.SetMinimumLevel(LogLevel.Warning)) | ||
| .ConfigureServices(services => | ||
| { | ||
| services.ConfigureDurableAgents( | ||
| options => options.AddAIAgent(agent, timeToLive: TimeSpan.FromHours(1)), | ||
| workerBuilder: builder => builder.UseDurableTaskScheduler(dtsConnectionString), | ||
| clientBuilder: builder => builder.UseDurableTaskScheduler(dtsConnectionString)); | ||
| }) | ||
| .Build(); | ||
|
|
||
| await host.StartAsync(); | ||
|
|
||
| // Get the agent proxy from services | ||
| IServiceProvider services = host.Services; | ||
| AIAgent agentProxy = services.GetRequiredKeyedService<AIAgent>(JokerName); | ||
|
|
||
| // Console colors for better UX | ||
| Console.ForegroundColor = ConsoleColor.Cyan; | ||
| Console.WriteLine("=== Single Agent Console Sample ==="); | ||
| Console.ResetColor(); | ||
| Console.WriteLine("Enter a message for the Joker agent (or 'exit' to quit):"); | ||
| Console.WriteLine(); | ||
|
|
||
| // Create a thread for the conversation | ||
| AgentThread thread = await agentProxy.GetNewThreadAsync(); | ||
|
|
||
| while (true) | ||
| { | ||
| // Read input from stdin | ||
| Console.ForegroundColor = ConsoleColor.Yellow; | ||
| Console.Write("You: "); | ||
| Console.ResetColor(); | ||
|
|
||
| string? input = Console.ReadLine(); | ||
| if (string.IsNullOrWhiteSpace(input) || input.Equals("exit", StringComparison.OrdinalIgnoreCase)) | ||
| { | ||
| break; | ||
| } | ||
|
|
||
| // Run the agent | ||
| Console.ForegroundColor = ConsoleColor.Green; | ||
| Console.Write("Joker: "); | ||
| Console.ResetColor(); | ||
|
|
||
| try | ||
| { | ||
| AgentResponse agentResponse = await agentProxy.RunAsync( | ||
| message: input, | ||
| thread: thread, | ||
| cancellationToken: CancellationToken.None); | ||
|
|
||
| Console.WriteLine(agentResponse.Text); | ||
| Console.WriteLine(); | ||
| } | ||
| catch (Exception ex) | ||
| { | ||
| Console.ForegroundColor = ConsoleColor.Red; | ||
| Console.Error.WriteLine($"Error: {ex.Message}"); | ||
| Console.ResetColor(); | ||
| Console.WriteLine(); | ||
| } | ||
| } | ||
|
|
||
| await host.StopAsync(); |
56 changes: 56 additions & 0 deletions
56
dotnet/samples/DurableAgents/ConsoleApps/01_SingleAgent/README.md
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| Original file line number | Diff line number | Diff line change |
|---|---|---|
| @@ -0,0 +1,56 @@ | ||
| # Single Agent Sample | ||
|
|
||
| This sample demonstrates how to use the durable agents extension to create a simple console app that hosts a single AI agent and provides interactive conversation via stdin/stdout. | ||
|
|
||
| ## Key Concepts Demonstrated | ||
|
|
||
| - Using the Microsoft Agent Framework to define a simple AI agent with a name and instructions. | ||
| - Registering durable agents with the console app and running them interactively. | ||
| - Conversation management (via threads) for isolated interactions. | ||
|
|
||
| ## Environment Setup | ||
|
|
||
| See the [README.md](../README.md) file in the parent directory for more information on how to configure the environment, including how to install and run common sample dependencies. | ||
|
|
||
| ## Running the Sample | ||
|
|
||
| With the environment setup, you can run the sample: | ||
|
|
||
| ```bash | ||
| cd dotnet/samples/DurableAgents/ConsoleApps/01_SingleAgent | ||
| dotnet run --framework net10.0 | ||
| ``` | ||
|
|
||
| The app will prompt you for input. You can interact with the Joker agent: | ||
|
|
||
| ```text | ||
| === Single Agent Console Sample === | ||
| Enter a message for the Joker agent (or 'exit' to quit): | ||
|
|
||
| You: Tell me a joke about a pirate. | ||
| Joker: Why don't pirates ever learn the alphabet? Because they always get stuck at "C"! | ||
|
|
||
| You: Now explain the joke. | ||
| Joker: The joke plays on the word "sea" (C), which pirates are famously associated with... | ||
|
|
||
| You: exit | ||
| ``` | ||
|
|
||
| ## Scriptable Usage | ||
|
|
||
| You can also pipe input to the app for scriptable usage: | ||
|
|
||
| ```bash | ||
| echo "Tell me a joke about a pirate." | dotnet run | ||
| ``` | ||
|
|
||
| The app will read from stdin, process the input, and write the response to stdout. | ||
|
|
||
| ## Viewing Agent State | ||
|
|
||
| You can view the state of the agent in the Durable Task Scheduler dashboard: | ||
|
|
||
| 1. Open your browser and navigate to `http://localhost:8082` | ||
| 2. In the dashboard, you can view the state of the Joker agent, including its conversation history and current state | ||
|
|
||
| The agent maintains conversation state across multiple interactions, and you can inspect this state in the dashboard to understand how the durable agents extension manages conversation context. |
30 changes: 30 additions & 0 deletions
30
...leAgents/ConsoleApps/02_AgentOrchestration_Chaining/02_AgentOrchestration_Chaining.csproj
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| Original file line number | Diff line number | Diff line change |
|---|---|---|
| @@ -0,0 +1,30 @@ | ||
| <Project Sdk="Microsoft.NET.Sdk"> | ||
| <PropertyGroup> | ||
| <TargetFrameworks>net10.0</TargetFrameworks> | ||
| <OutputType>Exe</OutputType> | ||
| <ImplicitUsings>enable</ImplicitUsings> | ||
| <Nullable>enable</Nullable> | ||
| <AssemblyName>AgentOrchestration_Chaining</AssemblyName> | ||
| <RootNamespace>AgentOrchestration_Chaining</RootNamespace> | ||
| </PropertyGroup> | ||
|
|
||
| <ItemGroup> | ||
| <PackageReference Include="Azure.AI.OpenAI" /> | ||
| <PackageReference Include="Azure.Identity" /> | ||
| <PackageReference Include="Microsoft.DurableTask.Client.AzureManaged" /> | ||
| <PackageReference Include="Microsoft.DurableTask.Worker.AzureManaged" /> | ||
| <PackageReference Include="Microsoft.Extensions.Hosting" /> | ||
| </ItemGroup> | ||
|
|
||
| <!-- Local projects that should be switched to package references when using the sample outside of this MAF repo --> | ||
| <!-- | ||
| <ItemGroup> | ||
| <PackageReference Include="Microsoft.Agents.AI.DurableTask" /> | ||
| <PackageReference Include="Microsoft.Agents.AI.OpenAI" /> | ||
| </ItemGroup> | ||
| --> | ||
| <ItemGroup> | ||
| <ProjectReference Include="..\..\..\..\src\Microsoft.Agents.AI.DurableTask\Microsoft.Agents.AI.DurableTask.csproj" /> | ||
| <ProjectReference Include="..\..\..\..\src\Microsoft.Agents.AI.OpenAI\Microsoft.Agents.AI.OpenAI.csproj" /> | ||
| </ItemGroup> | ||
| </Project> |
6 changes: 6 additions & 0 deletions
6
dotnet/samples/DurableAgents/ConsoleApps/02_AgentOrchestration_Chaining/Models.cs
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| Original file line number | Diff line number | Diff line change |
|---|---|---|
| @@ -0,0 +1,6 @@ | ||
| // Copyright (c) Microsoft. All rights reserved. | ||
|
|
||
| namespace AgentOrchestration_Chaining; | ||
|
|
||
| // Response model | ||
| public sealed record TextResponse(string Text); |
Oops, something went wrong.
Oops, something went wrong.
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
Uh oh!
There was an error while loading. Please reload this page.