122 lines
4.2 KiB
C#
Raw Normal View History

2025-01-27 13:00:05 -05:00
// Copyright (c) Microsoft Corporation. All rights reserved.
2025-01-27 13:05:25 -05:00
// BaseAgent.cs
2025-01-27 13:00:05 -05:00
using System.Diagnostics;
using System.Reflection;
using Microsoft.AutoGen.Contracts.Python;
using Microsoft.Extensions.Logging;
namespace Microsoft.AutoGen.Core.Python;
/// <summary>
/// Represents the base class for an agent in the AutoGen system.
/// </summary>
public abstract class BaseAgent : IAgent, IHostableAgent
{
/// <summary>
/// The activity source for tracing.
/// </summary>
public static readonly ActivitySource s_source = new("Microsoft.AutoGen.Core.Agent");
/// <summary>
/// Gets the unique identifier of the agent.
/// </summary>
public AgentId Id { get; private set; }
2025-01-27 14:35:19 -05:00
protected internal ILogger<BaseAgent> _logger;
2025-01-27 13:00:05 -05:00
public Type[] HandledTypes {
get {
return _handlersByMessageType.Keys.ToArray();
}
}
protected IUnboundSubscriptionDefinition[] GetUnboundSubscriptions()
{
throw new NotImplementedException();
}
protected IAgentRuntime Runtime { get; private set; }
private readonly Dictionary<Type, MethodInfo> _handlersByMessageType;
protected string Description { get; private set; }
public AgentMetadata Metadata {
get {
return new AgentMetadata
{
Type = Id.Type,
Key = Id.Key,
Description = Description
};
}
}
protected BaseAgent(
AgentId id,
IAgentRuntime runtime,
string description,
2025-01-27 14:35:19 -05:00
ILogger<BaseAgent>? logger = null)
2025-01-27 13:00:05 -05:00
{
Id = id;
2025-01-27 14:35:19 -05:00
_logger = logger ?? LoggerFactory.Create(builder => { }).CreateLogger<BaseAgent>();
2025-01-27 13:00:05 -05:00
Description = description;
Runtime = runtime;
_handlersByMessageType = new(GetType().GetHandlersLookupTable());
}
public ValueTask<object?> OnMessageAsync(object message, MessageContext messageContext)
{
// Determine type of message, then get handler method and invoke it
var messageType = message.GetType();
if (_handlersByMessageType.TryGetValue(messageType, out var handlerMethod))
{
// Determine if this is a IHandle<T> or IHandle<T, U> method
2025-01-27 14:35:19 -05:00
var genericArguments = handlerMethod.GetParameters();
2025-01-27 13:00:05 -05:00
if (genericArguments.Length == 1)
{
// This is a IHandle<T> method
var return_value = handlerMethod.Invoke(this, new object[] { message, messageContext });
return ValueTask.FromResult(return_value);
}
else if (genericArguments.Length == 2)
{
// This is a IHandle<T, U> method
// var _messageType = genericArguments[0];
// var _returnType = genericArguments[1];
var result = handlerMethod.Invoke(this, new object[] { message, messageContext });
return ValueTask.FromResult(result);
}
else
{
return ValueTask.FromException<object?>(new InvalidOperationException($"Unexpected number of generic arguments in handler method {handlerMethod.Name}"));
}
}
else
{
return ValueTask.FromException<object?>(new InvalidOperationException($"No handler found for message type {messageType.FullName}"));
}
}
2025-01-27 14:35:19 -05:00
public virtual ValueTask<IDictionary<string, object>> SaveStateAsync()
2025-01-27 13:00:05 -05:00
{
2025-01-27 14:35:19 -05:00
return ValueTask.FromResult<IDictionary<string, object>>(new Dictionary<string, object>());
2025-01-27 13:00:05 -05:00
}
2025-01-27 14:35:19 -05:00
public virtual ValueTask LoadStateAsync(IDictionary<string, object> state)
2025-01-27 13:00:05 -05:00
{
2025-01-27 14:35:19 -05:00
return ValueTask.CompletedTask;
2025-01-27 13:00:05 -05:00
}
public ValueTask<object?> SendMessageAsync(object message, AgentId recepient, string? messageId = null, CancellationToken? cancellationToken = default)
2025-01-27 13:00:05 -05:00
{
return this.Runtime.SendMessageAsync(message, recepient, sender: this.Id, messageId: messageId, cancellationToken: cancellationToken);
}
public ValueTask PublishMessageAsync(object message, TopicId topic, string? messageId = null, CancellationToken? cancellationToken = default)
2025-01-27 13:00:05 -05:00
{
return this.Runtime.PublishMessageAsync(message, topic, sender: this.Id, messageId: messageId, cancellationToken: cancellationToken);
}
}