mirror of
https://github.com/microsoft/agent-framework.git
synced 2026-06-16 21:04:09 +08:00
628bb1af48
* Update Foundry Responses as ChatClientAgent * Migrate obsolete AzureAI integration tests to versioned agent pattern Replace obsolete CreateAIAgentAsync/GetAIAgentAsync calls with Agents.CreateAgentVersionAsync() + AsAIAgent(AgentVersion) in all AzureAI integration tests. - Rename AIProjectClient* test files to FoundryVersionedAgent* - Register AIFunction tools in PromptAgentDefinition.Tools for server-side visibility via AsOpenAIResponseTool() - Skip structured output tests (AzureAIProjectChatClient clears ResponseFormat for versioned agents) - Remove all [Obsolete] attributes and #pragma warning disable CS0618 * Merge FoundryMemory package into AzureAI under Memory/ folder Move all FoundryMemory source, unit tests, and integration tests into the Microsoft.Agents.AI.AzureAI package. Change namespace from Microsoft.Agents.AI.FoundryMemory to Microsoft.Agents.AI.AzureAI. - Add [Experimental] to FoundryMemoryProviderOptions and Scope - Rename internal AIProjectClientExtensions to MemoryStoreExtensions - Update AzureAI .csproj with Compliance.Abstractions, Redaction - Remove FoundryMemory from solution and release filter - Update sample to reference AzureAI instead of FoundryMemory - Delete old Microsoft.Agents.AI.FoundryMemory project and tests * Add EnsureMemoryStoreCreatedAsync and memory existence checks to integration tests - Ensure memory store is created before testing memory operations - Add AZURE_AI_EMBEDDING_DEPLOYMENT_NAME config setting - Assert memories exist in store via SearchMemoriesAsync before cleanup - Verify scope isolation with direct memory store queries * Fix and rename AzureAI unit tests for RAPI vs Versioned clarity - Rename AsAIAgentAsync_* to AsAIAgent_* (drop Async from method group) - Add _Rapi_ prefix to non-versioned (Responses API) tests - Add _Versioned_ prefix to versioned agent tests where needed - Fix RAPI tests: assert GetService<AIProjectClient>() is null - Fix Versioned tests: assert IsType<FoundryAgent> and GetService<AIProjectClient>() returns the client instance - Fix UserAgent header tests: proper HTTP handler routing - Fix ChatClient_UsesDefaultConversationIdAsync test setup - All 153 unit tests pass with 0 failures * Rename Microsoft.Agents.AI.AzureAI to Microsoft.Agents.AI.Foundry Rename the project, namespace, folder, and all references from Microsoft.Agents.AI.AzureAI to Microsoft.Agents.AI.Foundry. Also rename Workflows.Declarative.AzureAI to .Foundry. - Rename src, unit test, integration test, and workflow folders - Update namespaces in all source and test .cs files - Update ProjectReferences in ~47 sample and test .csproj files - Update solution files (.slnx, .slnf) - Update sample using statements - Update READMEs, SKILL.md, ADRs in docs/ - Disable package validation baseline for renamed packages - Fix UTF-8 BOM encoding on all affected .cs files - AzureAI.Persistent left completely unchanged * Fix format: remove ImplicitUsings, add explicit usings, fix BOM encoding - Remove ImplicitUsings=enable from Foundry csproj to resolve IDE0005 on shared ReplacingRedactor.cs - Add explicit System usings to all source files that relied on them - Sort usings alphabetically per editorconfig rules - Fix UTF-8 BOM on 12 sample Program.cs files - Rename Azure AI Foundry Agents to Microsoft Foundry Agents in docs
191 lines
7.4 KiB
C#
191 lines
7.4 KiB
C#
// Copyright (c) Microsoft. All rights reserved.
|
|
|
|
using System;
|
|
using System.Threading.Tasks;
|
|
using Azure.AI.Projects;
|
|
using Azure.Identity;
|
|
using Microsoft.Agents.AI;
|
|
using Microsoft.Agents.AI.Foundry;
|
|
using Microsoft.Extensions.AI;
|
|
using Microsoft.Extensions.Configuration;
|
|
using OpenAI.Responses;
|
|
using Shared.IntegrationTests;
|
|
|
|
namespace Foundry.IntegrationTests.Memory;
|
|
|
|
/// <summary>
|
|
/// Integration tests for <see cref="FoundryMemoryProvider"/> against a configured Azure AI Foundry Memory service.
|
|
/// </summary>
|
|
/// <remarks>
|
|
/// These integration tests are skipped by default and require a live Azure AI Foundry Memory service.
|
|
/// The tests need to be updated to use the new AIAgent-based API pattern.
|
|
/// </remarks>
|
|
public sealed class FoundryMemoryProviderTests : IDisposable
|
|
{
|
|
private const string SkipReason = "Requires an Azure AI Foundry Memory service configured"; // Set to null to enable.
|
|
|
|
private readonly AIProjectClient? _client;
|
|
private readonly string? _memoryStoreName;
|
|
private readonly string? _deploymentName;
|
|
private readonly string? _embeddingDeploymentName;
|
|
private bool _disposed;
|
|
|
|
public FoundryMemoryProviderTests()
|
|
{
|
|
IConfigurationRoot configuration = new ConfigurationBuilder()
|
|
.AddJsonFile(path: "testsettings.development.json", optional: true, reloadOnChange: true)
|
|
.AddEnvironmentVariables()
|
|
.AddUserSecrets<FoundryMemoryProviderTests>(optional: true)
|
|
.Build();
|
|
|
|
var endpoint = configuration[TestSettings.AzureAIProjectEndpoint];
|
|
var memoryStoreName = configuration[TestSettings.AzureAIMemoryStoreId];
|
|
var deploymentName = configuration[TestSettings.AzureAIModelDeploymentName];
|
|
var embeddingDeploymentName = configuration[TestSettings.AzureAIEmbeddingDeploymentName];
|
|
|
|
if (!string.IsNullOrWhiteSpace(endpoint) &&
|
|
!string.IsNullOrWhiteSpace(memoryStoreName))
|
|
{
|
|
this._client = new AIProjectClient(new Uri(endpoint), new DefaultAzureCredential());
|
|
this._memoryStoreName = memoryStoreName;
|
|
this._deploymentName = deploymentName ?? "gpt-4.1-mini";
|
|
this._embeddingDeploymentName = embeddingDeploymentName ?? "text-embedding-ada-002";
|
|
}
|
|
}
|
|
|
|
[Fact(Skip = SkipReason)]
|
|
public async Task CanAddAndRetrieveUserMemoriesAsync()
|
|
{
|
|
// Arrange
|
|
FoundryMemoryProvider memoryProvider = new(
|
|
this._client!,
|
|
this._memoryStoreName!,
|
|
stateInitializer: _ => new(new FoundryMemoryProviderScope("it-user-1")));
|
|
|
|
await memoryProvider.EnsureMemoryStoreCreatedAsync(this._deploymentName!, this._embeddingDeploymentName!);
|
|
|
|
AIAgent agent = this._client!.AsAIAgent(new ChatClientAgentOptions
|
|
{
|
|
ChatOptions = new ChatOptions
|
|
{
|
|
ModelId = this._deploymentName!,
|
|
Instructions = "You are a helpful assistant. Use known memories about the user when responding, and do not invent details."
|
|
},
|
|
AIContextProviders = [memoryProvider]
|
|
});
|
|
|
|
AgentSession session = await agent.CreateSessionAsync();
|
|
|
|
await memoryProvider.EnsureStoredMemoriesDeletedAsync(session);
|
|
|
|
// Act
|
|
AgentResponse resultBefore = await agent.RunAsync("What is my name?", session);
|
|
Assert.DoesNotContain("Caoimhe", resultBefore.Text);
|
|
|
|
await agent.RunAsync("Hello, my name is Caoimhe.", session);
|
|
await memoryProvider.WhenUpdatesCompletedAsync();
|
|
await Task.Delay(2000);
|
|
|
|
// Assert - verify memories were actually created in the store before querying via agent
|
|
var searchResult = await this._client!.MemoryStores.SearchMemoriesAsync(
|
|
this._memoryStoreName!,
|
|
new MemorySearchOptions("it-user-1")
|
|
{
|
|
Items = { ResponseItem.CreateUserMessageItem("Caoimhe") }
|
|
});
|
|
Assert.NotEmpty(searchResult.Value.Memories);
|
|
|
|
AgentResponse resultAfter = await agent.RunAsync("What is my name?", session);
|
|
|
|
// Cleanup
|
|
await memoryProvider.EnsureStoredMemoriesDeletedAsync(session);
|
|
|
|
// Assert
|
|
Assert.Contains("Caoimhe", resultAfter.Text);
|
|
}
|
|
|
|
[Fact(Skip = SkipReason)]
|
|
public async Task DoesNotLeakMemoriesAcrossScopesAsync()
|
|
{
|
|
// Arrange
|
|
FoundryMemoryProvider memoryProvider1 = new(
|
|
this._client!,
|
|
this._memoryStoreName!,
|
|
stateInitializer: _ => new(new FoundryMemoryProviderScope("it-scope-a")));
|
|
|
|
FoundryMemoryProvider memoryProvider2 = new(
|
|
this._client!,
|
|
this._memoryStoreName!,
|
|
stateInitializer: _ => new(new FoundryMemoryProviderScope("it-scope-b")));
|
|
|
|
await memoryProvider1.EnsureMemoryStoreCreatedAsync(this._deploymentName!, this._embeddingDeploymentName!);
|
|
|
|
AIAgent agent1 = this._client!.AsAIAgent(new ChatClientAgentOptions
|
|
{
|
|
ChatOptions = new ChatOptions
|
|
{
|
|
ModelId = this._deploymentName!,
|
|
Instructions = "You are a helpful assistant. Use known memories about the user when responding, and do not invent details."
|
|
},
|
|
AIContextProviders = [memoryProvider1]
|
|
});
|
|
|
|
AIAgent agent2 = this._client!.AsAIAgent(new ChatClientAgentOptions
|
|
{
|
|
ChatOptions = new ChatOptions
|
|
{
|
|
ModelId = this._deploymentName!,
|
|
Instructions = "You are a helpful assistant. Use known memories about the user when responding, and do not invent details."
|
|
},
|
|
AIContextProviders = [memoryProvider2]
|
|
});
|
|
|
|
AgentSession session1 = await agent1.CreateSessionAsync();
|
|
AgentSession session2 = await agent2.CreateSessionAsync();
|
|
|
|
await memoryProvider1.EnsureStoredMemoriesDeletedAsync(session1);
|
|
await memoryProvider2.EnsureStoredMemoriesDeletedAsync(session2);
|
|
|
|
// Act - add memory only to scope A
|
|
await agent1.RunAsync("Hello, I'm an AI tutor and my name is Caoimhe.", session1);
|
|
await memoryProvider1.WhenUpdatesCompletedAsync();
|
|
await Task.Delay(2000);
|
|
|
|
// Assert - verify memories were created in scope A but not in scope B
|
|
var searchResultA = await this._client!.MemoryStores.SearchMemoriesAsync(
|
|
this._memoryStoreName!,
|
|
new MemorySearchOptions("it-scope-a")
|
|
{
|
|
Items = { ResponseItem.CreateUserMessageItem("Caoimhe") }
|
|
});
|
|
Assert.NotEmpty(searchResultA.Value.Memories);
|
|
|
|
var searchResultB = await this._client.MemoryStores.SearchMemoriesAsync(
|
|
this._memoryStoreName!,
|
|
new MemorySearchOptions("it-scope-b")
|
|
{
|
|
Items = { ResponseItem.CreateUserMessageItem("Caoimhe") }
|
|
});
|
|
Assert.Empty(searchResultB.Value.Memories);
|
|
|
|
AgentResponse result1 = await agent1.RunAsync("What is my name?", session1);
|
|
AgentResponse result2 = await agent2.RunAsync("What is my name?", session2);
|
|
|
|
// Assert
|
|
Assert.Contains("Caoimhe", result1.Text);
|
|
Assert.DoesNotContain("Caoimhe", result2.Text);
|
|
|
|
// Cleanup
|
|
await memoryProvider1.EnsureStoredMemoriesDeletedAsync(session1);
|
|
await memoryProvider2.EnsureStoredMemoriesDeletedAsync(session2);
|
|
}
|
|
|
|
public void Dispose()
|
|
{
|
|
if (!this._disposed)
|
|
{
|
|
this._disposed = true;
|
|
}
|
|
}
|
|
}
|