CODEX_ADK/BACKEND/Codex.CQRS/Queries/GetConversationQuery.cs
Svrnty 229a0698a3 Initial commit: CODEX_ADK monorepo
Multi-agent AI laboratory with ASP.NET Core 8.0 backend and Flutter frontend.
Implements CQRS architecture, OpenAPI contract-first API design.

BACKEND: Agent management, conversations, executions with PostgreSQL + Ollama
FRONTEND: Cross-platform UI with strict typing and Result-based error handling

Co-Authored-By: Jean-Philippe Brule <jp@svrnty.io>
2025-10-26 23:12:32 -04:00

125 lines
3.8 KiB
C#

using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading;
using System.Threading.Tasks;
using Codex.Dal;
using Codex.Dal.Enums;
using Microsoft.EntityFrameworkCore;
using OpenHarbor.CQRS.Abstractions;
namespace Codex.CQRS.Queries;
/// <summary>
/// Get conversation with all messages by ID
/// </summary>
public record GetConversationQuery
{
/// <summary>Conversation ID</summary>
public Guid Id { get; init; }
}
/// <summary>
/// Detailed conversation information with messages
/// </summary>
public record ConversationDetails
{
/// <summary>Unique conversation identifier</summary>
public Guid Id { get; init; }
/// <summary>Conversation title</summary>
public string Title { get; init; } = string.Empty;
/// <summary>Conversation summary</summary>
public string? Summary { get; init; }
/// <summary>Whether conversation is active</summary>
public bool IsActive { get; init; }
/// <summary>Conversation start timestamp</summary>
public DateTime StartedAt { get; init; }
/// <summary>Last message timestamp</summary>
public DateTime LastMessageAt { get; init; }
/// <summary>Total message count</summary>
public int MessageCount { get; init; }
/// <summary>All messages in conversation</summary>
public List<ConversationMessageItem> Messages { get; init; } = new();
}
/// <summary>
/// Individual message within a conversation
/// </summary>
public record ConversationMessageItem
{
/// <summary>Message identifier</summary>
public Guid Id { get; init; }
/// <summary>Conversation identifier</summary>
public Guid ConversationId { get; init; }
/// <summary>Execution identifier if from agent execution</summary>
public Guid? ExecutionId { get; init; }
/// <summary>Message role (user, assistant, system, tool)</summary>
public MessageRole Role { get; init; }
/// <summary>Message content</summary>
public string Content { get; init; } = string.Empty;
/// <summary>Message index/order in conversation</summary>
public int MessageIndex { get; init; }
/// <summary>Whether message is in active context window</summary>
public bool IsInActiveWindow { get; init; }
/// <summary>Message creation timestamp</summary>
public DateTime CreatedAt { get; init; }
}
public class GetConversationQueryHandler : IQueryHandler<GetConversationQuery, ConversationDetails?>
{
private readonly CodexDbContext _dbContext;
public GetConversationQueryHandler(CodexDbContext dbContext)
{
_dbContext = dbContext;
}
public async Task<ConversationDetails?> HandleAsync(
GetConversationQuery query,
CancellationToken cancellationToken = default)
{
return await _dbContext.Conversations
.AsNoTracking()
.Where(c => c.Id == query.Id)
.Select(c => new ConversationDetails
{
Id = c.Id,
Title = c.Title,
Summary = c.Summary,
IsActive = c.IsActive,
StartedAt = c.StartedAt,
LastMessageAt = c.LastMessageAt,
MessageCount = c.MessageCount,
Messages = c.Messages
.OrderBy(m => m.MessageIndex)
.Select(m => new ConversationMessageItem
{
Id = m.Id,
ConversationId = m.ConversationId,
ExecutionId = m.ExecutionId,
Role = m.Role,
Content = m.Content,
MessageIndex = m.MessageIndex,
IsInActiveWindow = m.IsInActiveWindow,
CreatedAt = m.CreatedAt
})
.ToList()
})
.FirstOrDefaultAsync(cancellationToken);
}
}