4ad0f9e493
## Backend — Service Layer & Repository Refactoring ### Neue Services (21 neue Dateien) **Interfaces & Implementierungen:** - `IOpenClawGatewayClient` — Interface für OpenClawGatewayClient (DIP-Fix: DashboardController hing an konkreter Klasse) - `IAgentConfigService` / `AgentConfigService` — Agent-Config-File-I/O aus AgentsController extrahiert - `IProjectService` / `ProjectService` — Projekt-CRUD + Activity-Logging (SRP) - `ITaskService` / `TaskService` — Task-State-Machine, Approve/Reject, Dashboard-Operationen (eliminiert Duplikation zwischen TasksController und DashboardController) - `IDashboardService` / `DashboardService` — Queue-Aggregation, Priority-Normalisierung, Gateway-Delegation - `IOperationsService` / `OperationsService` — Metriken-Berechnung aus OperationsController - `ITeamService` / `TeamService` — IDENTITY.md-Lesen aus TeamController - `IMemoryService` / `MemoryService` — File-I/O aus MemoryController - `IIncidentService` / `IncidentService` — File-Parsing (Regex-Source-Generatoren) aus IncidentsController - `IDocService` / `DocService` — Directory-Scan aus DocsController - `ICalendarService` / `CalendarService` — Gateway-HTTP-Calls + Fallback-Daten aus CalendarController ### Repository-Fixes **IUserRepository / UserRepository:** - `SaveChangesAsync` entfernt (leaky abstraction — Caller sollten nie SaveChanges steuern) - `RevokeTokenAsync(tokenHash)` — atomares Token-Revoke inkl. SaveChanges - `RevokeFamilyAsync(familyId)` — Batch-Revoke einer Token-Familie inkl. SaveChanges - `RemoveExpiredTokensAsync` speichert jetzt selbst (war vorher dependent auf nachfolgenden Save) ### AuthService-Fixes - `GetUserAsync`: unnötiges `Task.Run` entfernt → direkt `_users.GetByIdAsync().AsTask()` - `RevokeAsync`: delegiert jetzt an `IUserRepository.RevokeTokenAsync` - `RefreshAsync`: Token-Reuse-Detection delegiert an `IUserRepository.RevokeFamilyAsync` ### Bug-Fix - `OpenClawGatewayClient.ReadAgentGoalAsync`: pre-existing `CS1656` behoben (`reader` war `using`-Variable und wurde neu zugewiesen — in `reader2` umbenannt) ### Controller (16 Stück — alle slim) Alle Controller reduziert auf: Input validieren → Service aufrufen → HTTP-Result zurückgeben. Kein Business-Logic, kein File-I/O, keine direkte Repository-Nutzung (außer AgentsController für Activity-Log). **Program.cs — neue Registrierungen:** - `AddHttpClient<IOpenClawGatewayClient, OpenClawGatewayClient>` (war vorher konkrete Klasse) - Scoped: IDashboardService, IProjectService, ITaskService, IOperationsService, ITeamService, ICalendarService - Singleton: IAgentConfigService, IMemoryService, IIncidentService, IDocService --- ## Frontend — Dashboard V2 Components **AgentDetailModal.vue, IrisChat.vue, TaskStrip.vue:** - V2 Design-System: Dark Space Theme, Glass-Panels, Gradient-Akzente - Stores (agents, chat, tasks) nutzen Service + Mapper-Pattern - NexusLayout, FlowBoard, Topbar — Layoutfixes für fullHeight-Route-Meta Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
101 lines
3.5 KiB
C#
101 lines
3.5 KiB
C#
using Nexus.Api.Helpers;
|
|
|
|
namespace Nexus.Api.Services;
|
|
|
|
public sealed class MemoryService : IMemoryService
|
|
{
|
|
private const string BasePath = "/mnt/workspace-iris/memory";
|
|
private const string LongTermPath = "/mnt/workspace-iris/MEMORY.md";
|
|
private const int MaxFileSize = 1_000_000;
|
|
private const int MaxFiles = 50;
|
|
|
|
public Task<IReadOnlyList<MemoryFileInfo>> GetAllAsync()
|
|
{
|
|
var files = new List<MemoryFileInfo>();
|
|
|
|
if (File.Exists(LongTermPath))
|
|
{
|
|
var fi = new FileInfo(LongTermPath);
|
|
files.Add(new MemoryFileInfo("MEMORY.md", "MEMORY.md", fi.Length, fi.LastWriteTimeUtc));
|
|
}
|
|
|
|
if (Directory.Exists(BasePath))
|
|
{
|
|
var memFiles = Directory.GetFiles(BasePath, "*.md")
|
|
.Select(f => new FileInfo(f))
|
|
.OrderByDescending(f => f.Name)
|
|
.Select(f => new MemoryFileInfo(
|
|
f.Name,
|
|
f.FullName.Replace(BasePath, "").TrimStart('/'),
|
|
f.Length,
|
|
f.LastWriteTimeUtc));
|
|
files.AddRange(memFiles);
|
|
}
|
|
|
|
return Task.FromResult<IReadOnlyList<MemoryFileInfo>>(files);
|
|
}
|
|
|
|
public async Task<IReadOnlyList<MemorySearchResult>> SearchAsync(string query)
|
|
{
|
|
var results = new List<MemorySearchResult>();
|
|
|
|
async Task SearchDir(string dir)
|
|
{
|
|
if (!Directory.Exists(dir)) return;
|
|
foreach (var file in Directory.GetFiles(dir, "*.md").Take(MaxFiles))
|
|
{
|
|
var fi = new FileInfo(file);
|
|
if (fi.Length > MaxFileSize) continue;
|
|
var content = await File.ReadAllTextAsync(file);
|
|
if (!content.Contains(query, StringComparison.OrdinalIgnoreCase)) continue;
|
|
|
|
var idx = content.IndexOf(query, StringComparison.OrdinalIgnoreCase);
|
|
var start = Math.Max(0, idx - 60);
|
|
var excerpt = (start > 0 ? "…" : "") + content.Substring(start, Math.Min(200, content.Length - start)) + "…";
|
|
results.Add(new MemorySearchResult(
|
|
Path.GetFileName(file),
|
|
file.Replace(BasePath, "").TrimStart('/'),
|
|
excerpt,
|
|
fi.Length));
|
|
}
|
|
}
|
|
|
|
await SearchDir(BasePath);
|
|
|
|
if (File.Exists(LongTermPath))
|
|
{
|
|
var content = await File.ReadAllTextAsync(LongTermPath);
|
|
if (content.Contains(query, StringComparison.OrdinalIgnoreCase))
|
|
{
|
|
var idx = content.IndexOf(query, StringComparison.OrdinalIgnoreCase);
|
|
var start = Math.Max(0, idx - 60);
|
|
var excerpt = (start > 0 ? "…" : "") + content.Substring(start, Math.Min(200, content.Length - start)) + "…";
|
|
results.Insert(0, new MemorySearchResult("MEMORY.md", "MEMORY.md", excerpt, content.Length));
|
|
}
|
|
}
|
|
|
|
return results;
|
|
}
|
|
|
|
public async Task<MemoryFileContent?> GetFileAsync(string name)
|
|
{
|
|
string? filePath;
|
|
|
|
if (name.Equals("MEMORY.md", StringComparison.OrdinalIgnoreCase))
|
|
{
|
|
filePath = LongTermPath;
|
|
}
|
|
else
|
|
{
|
|
if (!PathSecurityHelper.TryResolveSafePath(BasePath, name, out filePath))
|
|
return null;
|
|
}
|
|
|
|
if (!File.Exists(filePath!))
|
|
return null;
|
|
|
|
var content = await File.ReadAllTextAsync(filePath!);
|
|
return new MemoryFileContent(name, name, content, content.Length, File.GetLastWriteTimeUtc(filePath!));
|
|
}
|
|
}
|