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>
90 lines
3.1 KiB
C#
90 lines
3.1 KiB
C#
using Nexus.Api.Helpers;
|
|
using System.Text.RegularExpressions;
|
|
|
|
namespace Nexus.Api.Services;
|
|
|
|
public sealed partial class IncidentService : IIncidentService
|
|
{
|
|
private const string BasePath = "/mnt/workspace-iris/memory/incidents";
|
|
|
|
public async Task<IReadOnlyList<IncidentSummary>> GetAllAsync()
|
|
{
|
|
if (!Directory.Exists(BasePath))
|
|
return Array.Empty<IncidentSummary>();
|
|
|
|
var incidents = new List<IncidentSummary>();
|
|
foreach (var file in Directory.GetFiles(BasePath, "*.md").OrderByDescending(f => f).Take(50))
|
|
{
|
|
var fi = new FileInfo(file);
|
|
if (fi.Length > 1_000_000) continue;
|
|
|
|
var name = Path.GetFileNameWithoutExtension(file);
|
|
var content = await File.ReadAllTextAsync(file);
|
|
var title = ExtractTitle(name, content);
|
|
var date = ExtractDate(name);
|
|
var severity = ExtractSeverity(content);
|
|
var excerpt = ExtractExcerpt(content);
|
|
|
|
incidents.Add(new IncidentSummary(Path.GetFileName(file), title, date, severity, excerpt, fi.Length));
|
|
}
|
|
|
|
return incidents;
|
|
}
|
|
|
|
public async Task<IncidentDetail?> GetByNameAsync(string name)
|
|
{
|
|
if (!PathSecurityHelper.TryResolveSafePath(BasePath, name, out var filePath))
|
|
return null;
|
|
|
|
if (!File.Exists(filePath!))
|
|
{
|
|
if (!name.EndsWith(".md", StringComparison.OrdinalIgnoreCase))
|
|
filePath = Path.Combine(BasePath, name + ".md");
|
|
if (!File.Exists(filePath!))
|
|
return null;
|
|
}
|
|
|
|
var content = await File.ReadAllTextAsync(filePath!);
|
|
var fi = new FileInfo(filePath!);
|
|
var fileName = Path.GetFileName(filePath!);
|
|
var title = ExtractTitle(Path.GetFileNameWithoutExtension(filePath!), content);
|
|
var date = ExtractDate(fileName);
|
|
|
|
return new IncidentDetail(fileName, title, date, content, fi.Length);
|
|
}
|
|
|
|
private static string ExtractTitle(string name, string content)
|
|
{
|
|
var match = TitleRegex().Match(content);
|
|
return match.Success ? match.Groups[1].Value.Trim() : name;
|
|
}
|
|
|
|
private static string? ExtractDate(string fileName)
|
|
{
|
|
var match = DateRegex().Match(fileName);
|
|
return match.Success ? match.Groups[1].Value : null;
|
|
}
|
|
|
|
private static string ExtractSeverity(string content)
|
|
{
|
|
var match = SeverityRegex().Match(content);
|
|
return match.Success ? match.Groups[1].Value.Trim() : "unknown";
|
|
}
|
|
|
|
private static string ExtractExcerpt(string content)
|
|
{
|
|
var excerptEnd = content.IndexOf("\n## ", StringComparison.Ordinal);
|
|
var excerpt = excerptEnd > 0 ? content[..excerptEnd].Trim() : content[..Math.Min(300, content.Length)].Trim();
|
|
return excerpt.Length > 200 ? excerpt[..200] + "…" : excerpt;
|
|
}
|
|
|
|
[GeneratedRegex(@"^#\s+(.+)$", RegexOptions.Multiline)]
|
|
private static partial Regex TitleRegex();
|
|
|
|
[GeneratedRegex(@"^(\d{4}-\d{2}-\d{2})")]
|
|
private static partial Regex DateRegex();
|
|
|
|
[GeneratedRegex(@"\*\*Severity:\*\*\s*(.+)$", RegexOptions.Multiline)]
|
|
private static partial Regex SeverityRegex();
|
|
}
|