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>
76 lines
2.4 KiB
C#
76 lines
2.4 KiB
C#
using Nexus.Api.Helpers;
|
|
|
|
namespace Nexus.Api.Services;
|
|
|
|
public sealed class DocService : IDocService
|
|
{
|
|
private static readonly string[] AllowedExtensions = [".md", ".json", ".txt", ".yaml", ".yml", ".html", ".css"];
|
|
private static readonly string[] SearchRoots =
|
|
[
|
|
"/mnt/workspace-iris",
|
|
"/home/node/.openclaw/workspace/nexus"
|
|
];
|
|
|
|
private static readonly (string Dir, string Category)[] ScanDirectories =
|
|
[
|
|
("/mnt/workspace-iris/nexus-phases", "phases"),
|
|
("/mnt/workspace-iris/skills", "skills"),
|
|
("/mnt/workspace-iris", "workspace"),
|
|
("/home/node/.openclaw/workspace/nexus", "nexus"),
|
|
("/home/node/.openclaw/workspace/nexus/phases", "nexus-phases")
|
|
];
|
|
|
|
public IReadOnlyList<DocFileInfo> GetAll()
|
|
{
|
|
var results = new List<DocFileInfo>();
|
|
|
|
foreach (var (dir, category) in ScanDirectories)
|
|
{
|
|
if (!Directory.Exists(dir)) continue;
|
|
foreach (var file in Directory.GetFiles(dir, "*.*"))
|
|
{
|
|
var ext = Path.GetExtension(file).ToLowerInvariant();
|
|
if (!AllowedExtensions.Contains(ext)) continue;
|
|
|
|
var fi = new FileInfo(file);
|
|
results.Add(new DocFileInfo(
|
|
fi.Name,
|
|
file.Replace("/mnt/workspace-iris", "").TrimStart('/'),
|
|
category,
|
|
ext.Replace(".", ""),
|
|
fi.Length,
|
|
fi.LastWriteTimeUtc));
|
|
}
|
|
}
|
|
|
|
return results.OrderByDescending(x => x.ModifiedAt).Take(100).ToList();
|
|
}
|
|
|
|
public async Task<DocFileContent?> GetFileAsync(string path)
|
|
{
|
|
if (string.IsNullOrWhiteSpace(path))
|
|
return null;
|
|
|
|
string? resolvedPath = null;
|
|
foreach (var root in SearchRoots)
|
|
{
|
|
if (PathSecurityHelper.TryResolveSafePath(root, path, out var candidate) && File.Exists(candidate))
|
|
{
|
|
resolvedPath = candidate;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (resolvedPath is null)
|
|
return null;
|
|
|
|
var content = await File.ReadAllTextAsync(resolvedPath);
|
|
var fi = new FileInfo(resolvedPath);
|
|
var relativePath = resolvedPath
|
|
.Replace("/mnt/workspace-iris/", "")
|
|
.Replace("/home/node/.openclaw/workspace/nexus/", "");
|
|
|
|
return new DocFileContent(fi.Name, relativePath, content, fi.Length, fi.LastWriteTimeUtc);
|
|
}
|
|
}
|