mirror of
https://github.com/microsoft/agent-framework.git
synced 2026-06-16 21:04:09 +08:00
aad20c2b33
* .NET: Bump Azure.AI.Projects to 2.1.0-beta.2 and add agent-endpoint AsAIAgent path
Bumps Azure.AI.Projects to 2.1.0-beta.2 with the matching transitive pins (Azure.Core 1.55.0, System.ClientModel 1.11.0).
Foundry agent endpoint plumbing:
* FoundryAgent now routes the agent-endpoint constructor through the new GetProjectResponsesClientForAgentEndpoint helper.
* Adds an internal FoundryAgent ctor that takes an existing AIProjectClient plus a parsed agent endpoint so the public extension does not need to construct a second project client.
* Adds public AIProjectClient.AsAIAgent(Uri agentEndpoint, ...) extension. This is the path consumer samples are expected to use for hosted agents because version selection happens server-side.
* Trims the dangling "If you want to construct a FoundryAgent against a project endpoint..." sentence from ParseAgentEndpoint.
Unit tests:
* Four new tests in AzureAIProjectChatClientExtensionsTests cover the AIProjectClient.AsAIAgent(Uri agentEndpoint, ...) overload. 263/263 pass.
Consumer samples (Using-Samples):
* SimpleAgent and SessionFilesClient now read AZURE_AI_PROJECT_ENDPOINT and AZURE_AI_AGENT_NAME (both required, throw on missing), derive the agent endpoint with new Uri($"{projectEndpoint}/agents/{agentName}/endpoint/protocols/openai"), then call aiProjectClient.AsAIAgent(agentEndpoint, ...).
* SessionFilesClient README updated.
Contributor samples (responses/*):
* New HostedContributorRouteExtensions.MapDevTemporaryLocalAgentEndpoint() wildcard route extension so localhost contributor servers accept the per-agent OpenAI endpoint shape the production Hosted runtime exposes.
* All 11 contributor Program.cs files call MapDevTemporaryLocalAgentEndpoint() with a contributor-only warning comment.
* Hosted-Files and Hosted-AzureSearchRag were importing Hosted_Shared_Contributor_Setup but never calling AddDevTemporaryLocalContributorSetup(). Both now call it so HostedSessionIsolationKeyProvider resolves correctly in dev.
* Hosted-AzureSearchRag, Hosted-Files, Hosted-MemoryAgent csprojs drop stale VersionOverride="2.1.0-beta.1" pins.
* Hosted-AzureSearchRag and Hosted-Files csprojs add ProjectReference to Hosted_Shared_Contributor_Setup.
* Hosted-Observability/.dockerignore removed the out/ exclusion that was blocking COPY out/ . in Dockerfile.contributor.
Verified:
* Full solution-scoped build of changed projects: green.
* Scoped CI-parity dotnet format via WSL2 + Docker (mcr.microsoft.com/dotnet/sdk:10.0) over every changed csproj: clean.
* Foundry unit tests: 263/263.
* Contributor docker smoke for 8 hosted samples (publish + docker build + docker run + curl POST to the wildcard route): HTTP 200 / 500 with route matched.
* End-to-end smoke against the real Azure Foundry project with a fresh bearer token: Hosted-Files contributor container served HTTP 200, the agent invoked ListBundledFiles, and returned the expected file name.
* Address PR review: forward pipeline settings; add UTs
- CreateProjectClientOptions also carries RetryPolicy, NetworkTimeout, ClientLoggingOptions, MessageLoggingPolicy (was Transport+UserAgentApplicationId only).
- Make CreateProjectClientOptions internal so tests can verify the copy directly.
- Add AsAIAgent(Uri) UTs covering tools forwarding to inner ChatOptions and null tools handling.
- Add CreateProjectClientOptions UTs covering null caller and full pipeline-settings copy.
131 lines
5.9 KiB
C#
131 lines
5.9 KiB
C#
// Copyright (c) Microsoft. All rights reserved.
|
|
|
|
// Seattle Hotel Agent - A hosted agent with local C# function tools.
|
|
// Demonstrates how to define and wire local tools that the LLM can invoke,
|
|
// a key advantage of code-based hosted agents over prompt agents.
|
|
|
|
using System.ComponentModel;
|
|
using System.Globalization;
|
|
using System.Text;
|
|
using Azure.AI.Projects;
|
|
using Azure.Core;
|
|
using Azure.Identity;
|
|
using DotNetEnv;
|
|
using Hosted_Shared_Contributor_Setup;
|
|
using Microsoft.Agents.AI;
|
|
using Microsoft.Agents.AI.Foundry.Hosting;
|
|
using Microsoft.Extensions.AI;
|
|
|
|
// Load .env file if present (for local development)
|
|
Env.TraversePath().Load();
|
|
|
|
string endpoint = Environment.GetEnvironmentVariable("AZURE_AI_PROJECT_ENDPOINT")
|
|
?? throw new InvalidOperationException("AZURE_AI_PROJECT_ENDPOINT is not set.");
|
|
string deploymentName = Environment.GetEnvironmentVariable("AZURE_AI_MODEL_DEPLOYMENT_NAME") ?? "gpt-4o";
|
|
|
|
// Use a chained credential: try a temporary dev token first (for local Docker debugging),
|
|
// then fall back to DefaultAzureCredential (for local dev via dotnet run / managed identity in production).
|
|
TokenCredential credential = new ChainedTokenCredential(
|
|
new DevTemporaryTokenCredential(),
|
|
new DefaultAzureCredential());
|
|
|
|
// ── Hotel data ───────────────────────────────────────────────────────────────
|
|
|
|
Hotel[] seattleHotels =
|
|
[
|
|
new("Contoso Suites", 189, 4.5, "Downtown"),
|
|
new("Fabrikam Residences", 159, 4.2, "Pike Place Market"),
|
|
new("Alpine Ski House", 249, 4.7, "Seattle Center"),
|
|
new("Margie's Travel Lodge", 219, 4.4, "Waterfront"),
|
|
new("Northwind Inn", 139, 4.0, "Capitol Hill"),
|
|
new("Relecloud Hotel", 99, 3.8, "University District"),
|
|
];
|
|
|
|
// ── Tool: GetAvailableHotels ─────────────────────────────────────────────────
|
|
|
|
[Description("Get available hotels in Seattle for the specified dates.")]
|
|
string GetAvailableHotels(
|
|
[Description("Check-in date in YYYY-MM-DD format")] string checkInDate,
|
|
[Description("Check-out date in YYYY-MM-DD format")] string checkOutDate,
|
|
[Description("Maximum price per night in USD (optional, defaults to 500)")] int maxPrice = 500)
|
|
{
|
|
if (!DateTime.TryParseExact(checkInDate, "yyyy-MM-dd", CultureInfo.InvariantCulture, DateTimeStyles.None, out var checkIn))
|
|
{
|
|
return "Error parsing check-in date. Please use YYYY-MM-DD format.";
|
|
}
|
|
|
|
if (!DateTime.TryParseExact(checkOutDate, "yyyy-MM-dd", CultureInfo.InvariantCulture, DateTimeStyles.None, out var checkOut))
|
|
{
|
|
return "Error parsing check-out date. Please use YYYY-MM-DD format.";
|
|
}
|
|
|
|
if (checkOut <= checkIn)
|
|
{
|
|
return "Error: Check-out date must be after check-in date.";
|
|
}
|
|
|
|
int nights = (checkOut - checkIn).Days;
|
|
List<Hotel> availableHotels = seattleHotels.Where(h => h.PricePerNight <= maxPrice).ToList();
|
|
|
|
if (availableHotels.Count == 0)
|
|
{
|
|
return $"No hotels found in Seattle within your budget of ${maxPrice}/night.";
|
|
}
|
|
|
|
StringBuilder result = new();
|
|
result.AppendLine($"Available hotels in Seattle from {checkInDate} to {checkOutDate} ({nights} nights):");
|
|
result.AppendLine();
|
|
|
|
foreach (Hotel hotel in availableHotels)
|
|
{
|
|
int totalCost = hotel.PricePerNight * nights;
|
|
result.AppendLine($"**{hotel.Name}**");
|
|
result.AppendLine($" Location: {hotel.Location}");
|
|
result.AppendLine($" Rating: {hotel.Rating}/5");
|
|
result.AppendLine($" ${hotel.PricePerNight}/night (Total: ${totalCost})");
|
|
result.AppendLine();
|
|
}
|
|
|
|
return result.ToString();
|
|
}
|
|
|
|
// ── Create and host the agent ────────────────────────────────────────────────
|
|
|
|
AIAgent agent = new AIProjectClient(new Uri(endpoint), credential)
|
|
.AsAIAgent(
|
|
model: deploymentName,
|
|
instructions: """
|
|
You are a helpful travel assistant specializing in finding hotels in Seattle, Washington.
|
|
|
|
When a user asks about hotels in Seattle:
|
|
1. Ask for their check-in and check-out dates if not provided
|
|
2. Ask about their budget preferences if not mentioned
|
|
3. Use the GetAvailableHotels tool to find available options
|
|
4. Present the results in a friendly, informative way
|
|
5. Offer to help with additional questions about the hotels or Seattle
|
|
|
|
Be conversational and helpful. If users ask about things outside of Seattle hotels,
|
|
politely let them know you specialize in Seattle hotel recommendations.
|
|
""",
|
|
name: Environment.GetEnvironmentVariable("AGENT_NAME") ?? "hosted-local-tools",
|
|
description: "Seattle hotel search agent with local function tools",
|
|
tools: [AIFunctionFactory.Create(GetAvailableHotels)]);
|
|
|
|
var builder = WebApplication.CreateBuilder(args);
|
|
builder.Services.AddFoundryResponses(agent);
|
|
builder.Services.AddDevTemporaryLocalContributorSetup(); // Local Docker debugging only - must not be used in production.
|
|
|
|
var app = builder.Build();
|
|
app.MapFoundryResponses();
|
|
|
|
// Contributor-only: in Development, also map the per-agent OpenAI route shape that live Foundry uses
|
|
// so a local REPL client can target this server via AIProjectClient.AsAIAgent(Uri agentEndpoint).
|
|
// Do not use this in production. Hosted Foundry agents only support the agent-endpoint path.
|
|
app.MapDevTemporaryLocalAgentEndpoint();
|
|
|
|
app.Run();
|
|
|
|
// ── Types ────────────────────────────────────────────────────────────────────
|
|
|
|
internal sealed record Hotel(string Name, int PricePerNight, double Rating, string Location);
|