Nota
L'accesso a questa pagina richiede l'autorizzazione. È possibile provare ad accedere o modificare le directory.
L'accesso a questa pagina richiede l'autorizzazione. È possibile provare a modificare le directory.
L'orchestrazione di chat di gruppo modella una conversazione collaborativa tra più agenti, coordinata da un agente di orchestrazione che determina la selezione dell'altoparlante e il flusso della conversazione. Questo modello è ideale per scenari che richiedono perfezionamento iterativo, risoluzione collaborativa dei problemi o analisi multi-prospettiva.
Internamente, l'orchestrazione della chat di gruppo dispone gli agenti in una topologia a stella, con un orchestratore al centro. L'agente di orchestrazione può implementare varie strategie per selezionare l'agente che parla successivamente, ad esempio round robin, selezione basata su prompt o logica personalizzata in base al contesto della conversazione, rendendolo un modello flessibile e potente per la collaborazione multi-agente.
Differenze tra chat di gruppo e altri modelli
L'orchestrazione di chat di gruppo presenta caratteristiche distinte rispetto ad altri modelli multi-agente:
- Coordinamento centralizzato: a differenza dei modelli di handoff in cui gli agenti trasferisce direttamente il controllo, la chat di gruppo usa un agente di orchestrazione per coordinare chi parla successivamente
- Perfezionamento iterativo: gli agenti possono esaminare e basarsi sulle risposte degli altri in più round
- Selezione voce flessibile: l'agente di orchestrazione può usare varie strategie (round robin, logica personalizzata e basata su prompt) per selezionare gli altoparlanti
- Contesto condiviso: tutti gli agenti visualizzano la cronologia completa della conversazione, abilitando il perfezionamento collaborativo
Contenuto dell'esercitazione
- Come creare agenti specializzati per la collaborazione di gruppi
- Come configurare le strategie di selezione voce
- Come creare flussi di lavoro con perfezionamento dell'agente iterativo
- Come personalizzare il flusso di conversazione con agenti di orchestrazione personalizzati
Configurare il client OpenAI di Azure
using System;
using System.Collections.Generic;
using System.Threading.Tasks;
using Azure.AI.OpenAI;
using Azure.Identity;
using Microsoft.Agents.AI.Workflows;
using Microsoft.Extensions.AI;
using Microsoft.Agents.AI;
// Set up the Azure OpenAI client
var endpoint = Environment.GetEnvironmentVariable("AZURE_OPENAI_ENDPOINT") ??
throw new InvalidOperationException("AZURE_OPENAI_ENDPOINT is not set.");
var deploymentName = Environment.GetEnvironmentVariable("AZURE_OPENAI_DEPLOYMENT_NAME") ?? "gpt-4o-mini";
var client = new AzureOpenAIClient(new Uri(endpoint), new AzureCliCredential())
.GetChatClient(deploymentName)
.AsIChatClient();
Definire gli agenti
Creare agenti specializzati per ruoli diversi nella conversazione di gruppo:
// Create a copywriter agent
ChatClientAgent writer = new(client,
"You are a creative copywriter. Generate catchy slogans and marketing copy. Be concise and impactful.",
"CopyWriter",
"A creative copywriter agent");
// Create a reviewer agent
ChatClientAgent reviewer = new(client,
"You are a marketing reviewer. Evaluate slogans for clarity, impact, and brand alignment. " +
"Provide constructive feedback or approval.",
"Reviewer",
"A marketing review agent");
Configurare la chat di gruppo con Round-Robin Orchestrator
Creare il flusso di lavoro di chat di gruppo usando AgentWorkflowBuilder:
// Build group chat with round-robin speaker selection
// The manager factory receives the list of agents and returns a configured manager
var workflow = AgentWorkflowBuilder
.CreateGroupChatBuilderWith(agents =>
new RoundRobinGroupChatManager(agents)
{
MaximumIterationCount = 5 // Maximum number of turns
})
.AddParticipants(writer, reviewer)
.Build();
Eseguire il flusso di lavoro di Chat di gruppo
Eseguire il flusso di lavoro e osservare la conversazione iterativa:
// Start the group chat
var messages = new List<ChatMessage> {
new(ChatRole.User, "Create a slogan for an eco-friendly electric vehicle.")
};
StreamingRun run = await InProcessExecution.StreamAsync(workflow, messages);
await run.TrySendMessageAsync(new TurnToken(emitEvents: true));
await foreach (WorkflowEvent evt in run.WatchStreamAsync().ConfigureAwait(false))
{
if (evt is AgentResponseUpdateEvent update)
{
// Process streaming agent responses
AgentResponse response = update.AsResponse();
foreach (ChatMessage message in response.Messages)
{
Console.WriteLine($"[{update.ExecutorId}]: {message.Text}");
}
}
else if (evt is WorkflowOutputEvent output)
{
// Workflow completed
var conversationHistory = output.As<List<ChatMessage>>();
Console.WriteLine("\n=== Final Conversation ===");
foreach (var message in conversationHistory)
{
Console.WriteLine($"{message.AuthorName}: {message.Text}");
}
break;
}
}
Interazione di esempio
[CopyWriter]: "Green Dreams, Zero Emissions" - Drive the future with style and sustainability.
[Reviewer]: The slogan is good, but "Green Dreams" might be a bit abstract. Consider something
more direct like "Pure Power, Zero Impact" to emphasize both performance and environmental benefit.
[CopyWriter]: "Pure Power, Zero Impact" - Experience electric excellence without compromise.
[Reviewer]: Excellent! This slogan is clear, impactful, and directly communicates the key benefits.
The tagline reinforces the message perfectly. Approved for use.
[CopyWriter]: Thank you! The final slogan is: "Pure Power, Zero Impact" - Experience electric
excellence without compromise.
Configurare il client chat
from agent_framework.azure import AzureOpenAIChatClient
from azure.identity import AzureCliCredential
# Initialize the Azure OpenAI chat client
chat_client = AzureOpenAIChatClient(credential=AzureCliCredential())
Definire gli agenti
Creare agenti specializzati con ruoli distinti:
from agent_framework import ChatAgent
# Create a researcher agent
researcher = ChatAgent(
name="Researcher",
description="Collects relevant background information.",
instructions="Gather concise facts that help answer the question. Be brief and factual.",
chat_client=chat_client,
)
# Create a writer agent
writer = ChatAgent(
name="Writer",
description="Synthesizes polished answers using gathered information.",
instructions="Compose clear, structured answers using any notes provided. Be comprehensive.",
chat_client=chat_client,
)
Configurare La chat di gruppo con selettore semplice
Creare una chat di gruppo con logica di selezione voce personalizzata:
from agent_framework import GroupChatBuilder, GroupChatState
def round_robin_selector(state: GroupChatState) -> str:
"""A round-robin selector function that picks the next speaker based on the current round index."""
participant_names = list(state.participants.keys())
return participant_names[state.current_round % len(participant_names)]
# Build the group chat workflow
workflow = (
GroupChatBuilder()
.with_orchestrator(selection_func=round_robin_selector)
.participants([researcher, writer])
# Terminate after 4 turns (researcher → writer → researcher → writer)
.with_termination_condition(lambda conversation: len(conversation) >= 4)
.build()
)
Configurare la chat di gruppo con orchestratore basato su agenti
In alternativa, usa un orchestratore basato su agenti per la selezione intelligente degli altoparlanti. L'orchestratore è completo ChatAgent con accesso a strumenti, contesto e osservabilità:
# Create orchestrator agent for speaker selection
orchestrator_agent = ChatAgent(
name="Orchestrator",
description="Coordinates multi-agent collaboration by selecting speakers",
instructions="""
You coordinate a team conversation to solve the user's task.
Guidelines:
- Start with Researcher to gather information
- Then have Writer synthesize the final answer
- Only finish after both have contributed meaningfully
""",
chat_client=chat_client,
)
# Build group chat with agent-based orchestrator
workflow = (
GroupChatBuilder()
.with_orchestrator(agent=orchestrator_agent)
# Set a hard termination condition: stop after 4 assistant messages
# The agent orchestrator will intelligently decide when to end before this limit but just in case
.with_termination_condition(lambda messages: sum(1 for msg in messages if msg.role == Role.ASSISTANT) >= 4)
.participants([researcher, writer])
.build()
)
Eseguire il flusso di lavoro di Chat di gruppo
Eseguire gli eventi del flusso di lavoro e del processo:
from typing import cast
from agent_framework import AgentResponseUpdateEvent, Role, WorkflowOutputEvent
task = "What are the key benefits of async/await in Python?"
print(f"Task: {task}\n")
print("=" * 80)
final_conversation: list[ChatMessage] = []
last_executor_id: str | None = None
# Run the workflow
async for event in workflow.run_stream(task):
if isinstance(event, AgentResponseUpdateEvent):
# Print streaming agent updates
eid = event.executor_id
if eid != last_executor_id:
if last_executor_id is not None:
print()
print(f"[{eid}]:", end=" ", flush=True)
last_executor_id = eid
print(event.data, end="", flush=True)
elif isinstance(event, WorkflowOutputEvent):
# Workflow completed - data is a list of ChatMessage
final_conversation = cast(list[ChatMessage], event.data)
if final_conversation:
print("\n\n" + "=" * 80)
print("Final Conversation:")
for msg in final_conversation:
author = getattr(msg, "author_name", "Unknown")
text = getattr(msg, "text", str(msg))
print(f"\n[{author}]\n{text}")
print("-" * 80)
print("\nWorkflow completed.")
Interazione di esempio
Task: What are the key benefits of async/await in Python?
================================================================================
[Researcher]: Async/await in Python provides non-blocking I/O operations, enabling
concurrent execution without threading overhead. Key benefits include improved
performance for I/O-bound tasks, better resource utilization, and simplified
concurrent code structure using native coroutines.
[Writer]: The key benefits of async/await in Python are:
1. **Non-blocking Operations**: Allows I/O operations to run concurrently without
blocking the main thread, significantly improving performance for network
requests, file I/O, and database queries.
2. **Resource Efficiency**: Avoids the overhead of thread creation and context
switching, making it more memory-efficient than traditional threading.
3. **Simplified Concurrency**: Provides a clean, synchronous-looking syntax for
asynchronous code, making concurrent programs easier to write and maintain.
4. **Scalability**: Enables handling thousands of concurrent connections with
minimal resource consumption, ideal for high-performance web servers and APIs.
--------------------------------------------------------------------------------
Workflow completed.
Concetti chiave
- Gestione centralizzata: la chat di gruppo usa un manager per coordinare la selezione e il flusso dell'altoparlante
- AgentWorkflowBuilder.CreateGroupChatBuilderWith(): crea flussi di lavoro con una funzione factory di gestione
- RoundRobinGroupChatManager: gestore predefinito che alterna gli altoparlanti in modalità round robin
- MaximumIterationCount: controlla il numero massimo di turni dell'agente prima della terminazione
-
Gestori personalizzati: estendere
RoundRobinGroupChatManagero implementare la logica personalizzata - Perfezionamento iterativo: gli agenti esaminano e migliorano i contributi degli altri
- Contesto condiviso: tutti i partecipanti visualizzano la cronologia completa della conversazione
-
Strategie dell'agente di orchestrazione flessibile: scegliere tra selettori semplici, agenti di orchestrazione basati su agenti o logica personalizzata usando
with_orchestrator(). - GroupChatBuilder: crea flussi di lavoro con selezione voce configurabile
- GroupChatState: fornisce lo stato della conversazione per le decisioni di selezione
- Collaborazione iterativa: gli agenti si basano sui contributi degli altri
-
Streaming di eventi: processa
AgentResponseUpdateEventeWorkflowOutputEventin tempo reale - list[ChatMessage] Output: Tutte le orchestrazioni restituiscono un elenco di messaggi di chat
Avanzate: selezione voce personalizzata
È possibile implementare la logica di gestione personalizzata creando un gestore chat di gruppo personalizzato:
public class ApprovalBasedManager : RoundRobinGroupChatManager
{
private readonly string _approverName;
public ApprovalBasedManager(IReadOnlyList<AIAgent> agents, string approverName)
: base(agents)
{
_approverName = approverName;
}
// Override to add custom termination logic
protected override ValueTask<bool> ShouldTerminateAsync(
IReadOnlyList<ChatMessage> history,
CancellationToken cancellationToken = default)
{
var last = history.LastOrDefault();
bool shouldTerminate = last?.AuthorName == _approverName &&
last.Text?.Contains("approve", StringComparison.OrdinalIgnoreCase) == true;
return ValueTask.FromResult(shouldTerminate);
}
}
// Use custom manager in workflow
var workflow = AgentWorkflowBuilder
.CreateGroupChatBuilderWith(agents =>
new ApprovalBasedManager(agents, "Reviewer")
{
MaximumIterationCount = 10
})
.AddParticipants(writer, reviewer)
.Build();
È possibile implementare una logica di selezione sofisticata in base allo stato della conversazione:
def smart_selector(state: GroupChatState) -> str:
"""Select speakers based on conversation content and context."""
conversation = state.conversation
last_message = conversation[-1] if conversation else None
# If no messages yet, start with Researcher
if not last_message:
return "Researcher"
# Check last message content
last_text = last_message.text.lower()
# If researcher finished gathering info, switch to writer
if "I have finished" in last_text and last_message.author_name == "Researcher":
return "Writer"
# Else continue with researcher until it indicates completion
return "Researcher"
workflow = (
GroupChatBuilder()
.with_orchestrator(selection_func=smart_selector, orchestrator_name="SmartOrchestrator")
.participants([researcher, writer])
.build()
)
Importante
Quando si usa un'implementazione personalizzata di BaseGroupChatOrchestrator per scenari avanzati, è necessario impostare tutte le proprietà, tra cui participant_registry, max_roundse termination_condition.
max_rounds e termination_condition impostati nel builder verranno ignorati.
Sincronizzazione del contesto
Come accennato all'inizio di questa guida, tutti gli agenti in una chat di gruppo vedono la cronologia completa della conversazione.
Gli agenti in Agent Framework si basano sulle sessioni dell'agente (AgentSession) per gestire il contesto. In un'orchestrazione di chat di gruppo gli agenti non condividono la stessa istanza di sessione, ma l'agente di orchestrazione garantisce che la sessione di ogni agente sia sincronizzata con la cronologia di conversazione completa prima di ogni turno. A tale scopo, dopo il turno di ogni agente, l'agente di orchestrazione trasmette la risposta a tutti gli altri agenti, assicurandosi che tutti i partecipanti abbiano il contesto più recente per il turno successivo.
Suggerimento
Gli agenti non condividono la stessa istanza di sessione perché diversi tipi di agenti possono avere implementazioni diverse dell'astrazione AgentSession . La condivisione della stessa istanza di sessione potrebbe causare incoerenze nel modo in cui ogni agente elabora e gestisce il contesto.
Dopo aver trasmesso la risposta, l'agente di orchestrazione decide quindi l'altoparlante successivo e invia una richiesta all'agente selezionato, che ora ha la cronologia completa della conversazione per generare la risposta.
Quando usare Chat di gruppo
L'orchestrazione di chat di gruppo è ideale per:
- Perfezionamento iterativo: più cicli di revisione e miglioramento
- Collaborative Problem-Solving: agenti con competenze complementari che interagiscono
- Creazione dei contenuti: Flussi di lavoro tra scrittore e revisore per la creazione di documenti
- Analisi multi-prospettiva: ottenere punti di vista diversi sullo stesso input
- Controllo qualità: processi automatizzati di revisione e approvazione
Prendere in considerazione le alternative quando:
- È necessaria un'elaborazione sequenziale rigorosa (usare l'orchestrazione sequenziale)
- Gli agenti devono funzionare in modo completamente indipendente (usare l'orchestrazione simultanea)
- Sono necessari trasferimenti diretti da agente ad agente (utilizzare l'orchestrazione dei trasferimenti)
- È necessaria una pianificazione dinamica complessa (usare l'orchestrazione magentica)