feat(search/rag): implement NexusSearchBox, dynamic Qdrant collection auto-provisioning, batch vector ingestion, mobile Serilog logging, and resolve 401 auth handler error (#51)
Resolves #52 This Pull Request introduces the **NexusSearchBox** search feature with premium unified styling, implements a robust **dynamic Qdrant collection auto-provisioning and batch-vector ingestion pipeline**, integrates a unified **Serilog logging infrastructure** for the Blazor Hybrid environment (MAUI), and resolves the **401 Unauthorized API header propagation error** inside mobile builds. ### 🚀 Key Implementations #### 1. Premium `NexusSearchBox` & Semantic Search UI * **NexusSearchBox Component:** Created an elegant search-as-you-type search box with smooth key navigation, quick-clearing, and seamless dynamic styling. * **Unified Aesthetics:** Refactored the search box isolated styling to align perfectly with the dashboard's design system using glassmorphism, `--nexus-neon` token gradients, and smooth pulse/fade animations. * **Semantic Search Integration:** Integrated semantic search query dispatching (`SearchLibrarySemanticallyQuery`) and wired up navigation seamlessly through the updated `ReaderNavigationService`. * **Tests Hardening:** Added/adapted query assertions in `QueryTests.cs` to guarantee safe parameterization and error boundary mapping. #### 2. Qdrant Collection Provisioning & Vector Ingestion * **Dynamic Auto-Provisioning:** Implemented dynamic checking and lazy-creation of the `knowledge_units` collection using 768 dimensions and Cosine distance. * **High-Performance Ingestion:** Optimized `ProcessKnowledgeUnitsAsync` with high-performance batch embedding generation using `_embeddingGenerator` and deterministic MD5 GUIDs for stable, duplicate-free upsertion. * **Database Cache Clear Sync:** Integrated Qdrant collection deletion in `ClearCacheAsync` to ensure absolute consistency between the PostgreSQL database cache and vector database indices. #### 3. Cross-Platform MAUI Logging (Serilog Infrastructure) * **Serilog Integration:** Configured cross-platform Serilog routing in `SerilogConfiguration.cs`, streaming diagnostic logs safely across native platforms and the Blazor Webview container. * **Interop Bridge:** Built `BlazorLoggingBridge.cs` to capture web console messages and pipe them directly to the native host logger. * **Demo Interface:** Added an interactive `SerilogDemo.razor` sandbox under Pages. #### 4. Resolving 401 Load Errors (Authentication Handler Flow) * **Authentication Header Handler:** Implemented the `MobileAuthenticationHeaderHandler` to correctly extract, validate, and inject bearer JWT tokens into outbound API requests. * **Configuration-based API Host:** Structured standard API URI routing to use clean configuration bindings in `appsettings.json`. --- ### 🧪 Verification & Build Status * Run `dotnet build` from the solution root: Successfully compiled the full multi-targeted solution (`Liczba błędów: 0`). * All unit and integration tests successfully executed and verified (`dotnet test`). --------- Co-authored-by: Marek Jasiński <jasins.marek@gmail.com> Co-authored-by: Marek Jaisński <jasins.marek@gmail.com> Reviewed-on: #51 Co-authored-by: Antigravity <antigravity@google.com> Co-committed-by: Antigravity <antigravity@google.com>
This commit was merged in pull request #51.
This commit is contained in:
@@ -23,6 +23,11 @@ public interface IEbookRepository
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
void AddEbook(Ebook ebook);
|
void AddEbook(Ebook ebook);
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Finds an ebook by its unique identifier.
|
||||||
|
/// </summary>
|
||||||
|
Task<Ebook?> FindByIdAsync(Guid id, CancellationToken cancellationToken = default);
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Persists all staged changes to the underlying store.
|
/// Persists all staged changes to the underlying store.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
|
|||||||
@@ -0,0 +1,25 @@
|
|||||||
|
using NexusReader.Domain.Entities;
|
||||||
|
|
||||||
|
namespace NexusReader.Application.Abstractions.Persistence;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Abstraction for QuizResult and related User entity lookup.
|
||||||
|
/// Defined in the Application layer to maintain Clean Architecture isolation.
|
||||||
|
/// </summary>
|
||||||
|
public interface IQuizResultRepository
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Finds a user by ID to extract tenant context.
|
||||||
|
/// </summary>
|
||||||
|
Task<NexusUser?> FindUserByIdAsync(string userId, CancellationToken cancellationToken = default);
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Adds a new quiz result to the database.
|
||||||
|
/// </summary>
|
||||||
|
void AddQuizResult(QuizResult quizResult);
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Persists all staged changes to the repository.
|
||||||
|
/// </summary>
|
||||||
|
Task<int> SaveChangesAsync(CancellationToken cancellationToken = default);
|
||||||
|
}
|
||||||
@@ -0,0 +1,17 @@
|
|||||||
|
using FluentResults;
|
||||||
|
|
||||||
|
namespace NexusReader.Application.Abstractions.Services;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Service abstraction to extract raw text content from EPUB chapters.
|
||||||
|
/// </summary>
|
||||||
|
public interface IEpubExtractor
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Extracts the sanitized, plain-text content of each chapter in the EPUB file.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="relativePath">The relative storage path of the EPUB file.</param>
|
||||||
|
/// <param name="cancellationToken">Cancellation token.</param>
|
||||||
|
/// <returns>A list of plain-text chapters, or a failure result.</returns>
|
||||||
|
Task<Result<List<string>>> ExtractChaptersTextAsync(string relativePath, CancellationToken cancellationToken = default);
|
||||||
|
}
|
||||||
@@ -11,4 +11,5 @@ public interface IIdentityService
|
|||||||
Task<Result> LogoutAsync();
|
Task<Result> LogoutAsync();
|
||||||
Task<Result<UserProfileDto>> GetProfileAsync();
|
Task<Result<UserProfileDto>> GetProfileAsync();
|
||||||
Task<Result> RefreshTokenAsync();
|
Task<Result> RefreshTokenAsync();
|
||||||
|
void ClearCache();
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,5 +1,8 @@
|
|||||||
using FluentResults;
|
using FluentResults;
|
||||||
|
using System.Linq;
|
||||||
using MediatR;
|
using MediatR;
|
||||||
|
using Microsoft.Extensions.DependencyInjection;
|
||||||
|
using Microsoft.Extensions.Logging;
|
||||||
using NexusReader.Application.Abstractions.Messaging;
|
using NexusReader.Application.Abstractions.Messaging;
|
||||||
using NexusReader.Application.Abstractions.Persistence;
|
using NexusReader.Application.Abstractions.Persistence;
|
||||||
using NexusReader.Application.Abstractions.Services;
|
using NexusReader.Application.Abstractions.Services;
|
||||||
@@ -11,13 +14,16 @@ public class IngestEbookCommandHandler : IRequestHandler<IngestEbookCommand, Res
|
|||||||
{
|
{
|
||||||
private readonly IEbookRepository _ebookRepository;
|
private readonly IEbookRepository _ebookRepository;
|
||||||
private readonly IBookStorageService _storageService;
|
private readonly IBookStorageService _storageService;
|
||||||
|
private readonly IServiceScopeFactory _scopeFactory;
|
||||||
|
|
||||||
public IngestEbookCommandHandler(
|
public IngestEbookCommandHandler(
|
||||||
IEbookRepository ebookRepository,
|
IEbookRepository ebookRepository,
|
||||||
IBookStorageService storageService)
|
IBookStorageService storageService,
|
||||||
|
IServiceScopeFactory scopeFactory)
|
||||||
{
|
{
|
||||||
_ebookRepository = ebookRepository;
|
_ebookRepository = ebookRepository;
|
||||||
_storageService = storageService;
|
_storageService = storageService;
|
||||||
|
_scopeFactory = scopeFactory;
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task<Result<Guid>> Handle(IngestEbookCommand request, CancellationToken cancellationToken)
|
public async Task<Result<Guid>> Handle(IngestEbookCommand request, CancellationToken cancellationToken)
|
||||||
@@ -72,6 +78,43 @@ public class IngestEbookCommandHandler : IRequestHandler<IngestEbookCommand, Res
|
|||||||
_ebookRepository.AddEbook(ebook);
|
_ebookRepository.AddEbook(ebook);
|
||||||
await _ebookRepository.SaveChangesAsync(cancellationToken);
|
await _ebookRepository.SaveChangesAsync(cancellationToken);
|
||||||
|
|
||||||
|
// 4. Trigger asynchronous background processing and vector indexing
|
||||||
|
_ = Task.Run(async () =>
|
||||||
|
{
|
||||||
|
using var scope = _scopeFactory.CreateScope();
|
||||||
|
var logger = scope.ServiceProvider.GetRequiredService<ILogger<IngestEbookCommandHandler>>();
|
||||||
|
var broadcaster = scope.ServiceProvider.GetRequiredService<ISyncBroadcaster>();
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var mediator = scope.ServiceProvider.GetRequiredService<IMediator>();
|
||||||
|
var result = await mediator.Send(new ProcessEbookCommand(ebook.Id, request.UserId, request.TenantId));
|
||||||
|
if (result.IsFailed)
|
||||||
|
{
|
||||||
|
var errorMsg = string.Join("; ", result.Errors.Select(e => e.Message));
|
||||||
|
logger.LogError("[IngestEbook] Background ebook processing failed for Ebook {EbookId}: {Error}", ebook.Id, errorMsg);
|
||||||
|
await broadcaster.BroadcastIngestionProgressAsync(
|
||||||
|
request.UserId,
|
||||||
|
$"Błąd indeksowania: {errorMsg}",
|
||||||
|
1.0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
logger.LogError(ex, "[IngestEbook] Exception during background ebook processing for Ebook {EbookId}", ebook.Id);
|
||||||
|
try
|
||||||
|
{
|
||||||
|
await broadcaster.BroadcastIngestionProgressAsync(
|
||||||
|
request.UserId,
|
||||||
|
$"Błąd krytyczny podczas przetwarzania e-booka: {ex.Message}",
|
||||||
|
1.0);
|
||||||
|
}
|
||||||
|
catch
|
||||||
|
{
|
||||||
|
// Ignore broadcast failures to prevent crashes
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
return Result.Ok(ebook.Id);
|
return Result.Ok(ebook.Id);
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
|
|||||||
@@ -0,0 +1,176 @@
|
|||||||
|
using System.Text.RegularExpressions;
|
||||||
|
using FluentResults;
|
||||||
|
using MediatR;
|
||||||
|
using Microsoft.EntityFrameworkCore;
|
||||||
|
using Microsoft.Extensions.DependencyInjection;
|
||||||
|
using Microsoft.Extensions.Logging;
|
||||||
|
using NexusReader.Application.Abstractions.Messaging;
|
||||||
|
using NexusReader.Application.Abstractions.Persistence;
|
||||||
|
using NexusReader.Application.Abstractions.Services;
|
||||||
|
|
||||||
|
namespace NexusReader.Application.Commands.Library;
|
||||||
|
|
||||||
|
public record ProcessEbookCommand(
|
||||||
|
Guid EbookId,
|
||||||
|
string UserId,
|
||||||
|
string TenantId
|
||||||
|
) : ICommand<bool>;
|
||||||
|
|
||||||
|
public class ProcessEbookCommandHandler : IRequestHandler<ProcessEbookCommand, Result<bool>>
|
||||||
|
{
|
||||||
|
private readonly IEbookRepository _ebookRepository;
|
||||||
|
private readonly IKnowledgeService _knowledgeService;
|
||||||
|
private readonly IEpubExtractor _epubExtractor;
|
||||||
|
private readonly ISyncBroadcaster _broadcaster;
|
||||||
|
private readonly ILogger<ProcessEbookCommandHandler> _logger;
|
||||||
|
|
||||||
|
public ProcessEbookCommandHandler(
|
||||||
|
IEbookRepository ebookRepository,
|
||||||
|
IKnowledgeService knowledgeService,
|
||||||
|
IEpubExtractor epubExtractor,
|
||||||
|
ISyncBroadcaster broadcaster,
|
||||||
|
ILogger<ProcessEbookCommandHandler> logger)
|
||||||
|
{
|
||||||
|
_ebookRepository = ebookRepository;
|
||||||
|
_knowledgeService = knowledgeService;
|
||||||
|
_epubExtractor = epubExtractor;
|
||||||
|
_broadcaster = broadcaster;
|
||||||
|
_logger = logger;
|
||||||
|
}
|
||||||
|
|
||||||
|
public async Task<Result<bool>> Handle(ProcessEbookCommand request, CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
_logger.LogInformation("[ProcessEbook] Starting background processing for Ebook: {EbookId}", request.EbookId);
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
await _broadcaster.BroadcastIngestionProgressAsync(request.UserId, "Wyszukiwanie e-booka w bazie danych...", 0.05, cancellationToken);
|
||||||
|
|
||||||
|
var ebook = await _ebookRepository.FindByIdAsync(request.EbookId, cancellationToken);
|
||||||
|
if (ebook == null)
|
||||||
|
{
|
||||||
|
_logger.LogError("[ProcessEbook] Ebook not found in database: {EbookId}", request.EbookId);
|
||||||
|
return Result.Fail<bool>($"Ebook nie znaleziony w bazie danych: {request.EbookId}");
|
||||||
|
}
|
||||||
|
|
||||||
|
_logger.LogInformation("[ProcessEbook] Extracting chapters text for Ebook: {Title} ({FilePath})", ebook.Title, ebook.FilePath);
|
||||||
|
await _broadcaster.BroadcastIngestionProgressAsync(request.UserId, "Otwieranie i parsowanie pliku EPUB...", 0.1, cancellationToken);
|
||||||
|
|
||||||
|
var extractionResult = await _epubExtractor.ExtractChaptersTextAsync(ebook.FilePath, cancellationToken);
|
||||||
|
if (extractionResult.IsFailed)
|
||||||
|
{
|
||||||
|
var errorMsg = extractionResult.Errors.FirstOrDefault()?.Message ?? "Failed to extract text chapters.";
|
||||||
|
_logger.LogError("[ProcessEbook] Extraction failed: {Error}", errorMsg);
|
||||||
|
return Result.Fail<bool>(extractionResult.Errors);
|
||||||
|
}
|
||||||
|
|
||||||
|
var chapters = extractionResult.Value;
|
||||||
|
if (chapters == null || !chapters.Any())
|
||||||
|
{
|
||||||
|
_logger.LogWarning("[ProcessEbook] EPUB has no readable content files: {EbookId}", request.EbookId);
|
||||||
|
return Result.Fail<bool>("EPUB nie zawiera czytelnych rozdziałów.");
|
||||||
|
}
|
||||||
|
|
||||||
|
int totalChapters = chapters.Count;
|
||||||
|
_logger.LogInformation("[ProcessEbook] Processing {Count} chapters for Ebook: {Title}", totalChapters, ebook.Title);
|
||||||
|
|
||||||
|
await _broadcaster.BroadcastIngestionProgressAsync(request.UserId, $"Analizowanie struktury ({totalChapters} rozdziałów)...", 0.15, cancellationToken);
|
||||||
|
|
||||||
|
int processedChapters = 0;
|
||||||
|
|
||||||
|
for (int i = 0; i < totalChapters; i++)
|
||||||
|
{
|
||||||
|
var cleanText = chapters[i];
|
||||||
|
|
||||||
|
if (cleanText.Length < 100)
|
||||||
|
{
|
||||||
|
_logger.LogInformation("[ProcessEbook] Skipping chapter {Index} (text too short: {Length} chars)", i, cleanText.Length);
|
||||||
|
processedChapters++;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Chunk the text to maintain granular Knowledge Units
|
||||||
|
var chunks = ChunkText(cleanText, 3000);
|
||||||
|
_logger.LogInformation("[ProcessEbook] Chapter {Index} split into {ChunkCount} chunk(s)", i, chunks.Count);
|
||||||
|
|
||||||
|
foreach (var chunk in chunks)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
// Invoke GetKnowledgeMapAsync to extract, embed, and upsert knowledge units
|
||||||
|
var result = await _knowledgeService.GetKnowledgeMapAsync(chunk, request.TenantId, request.EbookId, cancellationToken);
|
||||||
|
if (result.IsFailed)
|
||||||
|
{
|
||||||
|
_logger.LogWarning("[ProcessEbook] Failed to generate knowledge map for a chunk of chapter {Index}: {Error}", i, result.Errors.FirstOrDefault()?.Message);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
_logger.LogError(ex, "[ProcessEbook] Exception during AI vectorization of chapter {Index} chunk", i);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
processedChapters++;
|
||||||
|
double progress = 0.15 + (0.75 * processedChapters / totalChapters);
|
||||||
|
await _broadcaster.BroadcastIngestionProgressAsync(
|
||||||
|
request.UserId,
|
||||||
|
$"Przetwarzanie rozdziału {processedChapters} z {totalChapters} przez AI...",
|
||||||
|
progress,
|
||||||
|
cancellationToken);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Mark the ebook as ready
|
||||||
|
ebook.IsReadyForReading = true;
|
||||||
|
await _ebookRepository.SaveChangesAsync(cancellationToken);
|
||||||
|
|
||||||
|
_logger.LogInformation("[ProcessEbook] Ingestion and vector indexing completed for: {Title}", ebook.Title);
|
||||||
|
|
||||||
|
await _broadcaster.BroadcastIngestionProgressAsync(
|
||||||
|
request.UserId,
|
||||||
|
"Indeksowanie wektorowe e-booka przez Nexus AI zakończone pomyślnie!",
|
||||||
|
1.0,
|
||||||
|
cancellationToken);
|
||||||
|
|
||||||
|
return Result.Ok(true);
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
_logger.LogError(ex, "[ProcessEbook] Critical error during background EPUB vectorization of ebook {EbookId}", request.EbookId);
|
||||||
|
await _broadcaster.BroadcastIngestionProgressAsync(
|
||||||
|
request.UserId,
|
||||||
|
$"Błąd indeksowania: {ex.Message}",
|
||||||
|
1.0,
|
||||||
|
cancellationToken);
|
||||||
|
return Result.Fail<bool>(new Error("Wystąpił błąd podczas indeksowania e-booka przez AI").CausedBy(ex));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static List<string> ChunkText(string text, int maxWords = 3000)
|
||||||
|
{
|
||||||
|
var words = text.Split(' ', StringSplitOptions.RemoveEmptyEntries);
|
||||||
|
var chunks = new List<string>();
|
||||||
|
if (words.Length <= maxWords)
|
||||||
|
{
|
||||||
|
chunks.Add(text);
|
||||||
|
return chunks;
|
||||||
|
}
|
||||||
|
var currentChunk = new List<string>();
|
||||||
|
int count = 0;
|
||||||
|
foreach (var word in words)
|
||||||
|
{
|
||||||
|
currentChunk.Add(word);
|
||||||
|
count++;
|
||||||
|
if (count >= maxWords)
|
||||||
|
{
|
||||||
|
chunks.Add(string.Join(" ", currentChunk));
|
||||||
|
currentChunk.Clear();
|
||||||
|
count = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (currentChunk.Any())
|
||||||
|
{
|
||||||
|
chunks.Add(string.Join(" ", currentChunk));
|
||||||
|
}
|
||||||
|
return chunks;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,10 @@
|
|||||||
|
using FluentResults;
|
||||||
|
using NexusReader.Application.Abstractions.Messaging;
|
||||||
|
|
||||||
|
namespace NexusReader.Application.Commands.Quiz;
|
||||||
|
|
||||||
|
public record SubmitQuizResultCommand(
|
||||||
|
string UserId,
|
||||||
|
string Topic,
|
||||||
|
int Score,
|
||||||
|
int TotalQuestions) : ICommand;
|
||||||
@@ -0,0 +1,41 @@
|
|||||||
|
using FluentResults;
|
||||||
|
using NexusReader.Application.Abstractions.Messaging;
|
||||||
|
using NexusReader.Application.Abstractions.Persistence;
|
||||||
|
using NexusReader.Domain.Entities;
|
||||||
|
|
||||||
|
namespace NexusReader.Application.Commands.Quiz;
|
||||||
|
|
||||||
|
public sealed class SubmitQuizResultCommandHandler : ICommandHandler<SubmitQuizResultCommand>
|
||||||
|
{
|
||||||
|
private readonly IQuizResultRepository _quizResultRepository;
|
||||||
|
|
||||||
|
public SubmitQuizResultCommandHandler(IQuizResultRepository quizResultRepository)
|
||||||
|
{
|
||||||
|
_quizResultRepository = quizResultRepository;
|
||||||
|
}
|
||||||
|
|
||||||
|
public async Task<Result> Handle(SubmitQuizResultCommand request, CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
var user = await _quizResultRepository.FindUserByIdAsync(request.UserId, cancellationToken);
|
||||||
|
if (user == null)
|
||||||
|
{
|
||||||
|
return Result.Fail("User not found.");
|
||||||
|
}
|
||||||
|
|
||||||
|
var quizResult = new QuizResult
|
||||||
|
{
|
||||||
|
Id = Guid.NewGuid(),
|
||||||
|
UserId = request.UserId,
|
||||||
|
TenantId = string.IsNullOrEmpty(user.TenantId) ? "global" : user.TenantId,
|
||||||
|
Topic = request.Topic,
|
||||||
|
Score = request.Score,
|
||||||
|
TotalQuestions = request.TotalQuestions,
|
||||||
|
CompletedDate = DateTime.UtcNow
|
||||||
|
};
|
||||||
|
|
||||||
|
_quizResultRepository.AddQuizResult(quizResult);
|
||||||
|
await _quizResultRepository.SaveChangesAsync(cancellationToken);
|
||||||
|
|
||||||
|
return Result.Ok();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -13,4 +13,6 @@ public class CitationDto
|
|||||||
public string CitationId { get; set; } = string.Empty; // e.g., chunk hash/ID
|
public string CitationId { get; set; } = string.Empty; // e.g., chunk hash/ID
|
||||||
public string Snippet { get; set; } = string.Empty; // Verified text snippet from context
|
public string Snippet { get; set; } = string.Empty; // Verified text snippet from context
|
||||||
public string SourceBook { get; set; } = string.Empty; // Book title or description
|
public string SourceBook { get; set; } = string.Empty; // Book title or description
|
||||||
|
public string? Author { get; set; }
|
||||||
|
public int? PageNumber { get; set; }
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ namespace NexusReader.Application.DTOs.User;
|
|||||||
public record UserProfileDto
|
public record UserProfileDto
|
||||||
{
|
{
|
||||||
public string Email { get; init; } = string.Empty;
|
public string Email { get; init; } = string.Empty;
|
||||||
|
public string UserId { get; init; } = string.Empty;
|
||||||
public int AITokensUsed { get; init; }
|
public int AITokensUsed { get; init; }
|
||||||
public Guid TenantId { get; init; }
|
public Guid TenantId { get; init; }
|
||||||
|
|
||||||
@@ -15,11 +16,12 @@ public record UserProfileDto
|
|||||||
|
|
||||||
public int AverageQuizScore { get; init; }
|
public int AverageQuizScore { get; init; }
|
||||||
|
|
||||||
/// <summary>
|
public string? DisplayName { get; init; }
|
||||||
/// Summary of the last read book.
|
public int BooksReadCount { get; init; }
|
||||||
/// </summary>
|
public int ConceptsMappedCount { get; init; }
|
||||||
public LastReadBookDto? LastReadBook { get; init; }
|
public LastReadBookDto? LastReadBook { get; init; }
|
||||||
|
public IReadOnlyList<QuizResultDto> RecentQuizzes { get; init; } = Array.Empty<QuizResultDto>();
|
||||||
|
public IReadOnlyList<MappedConceptDto> MappedConcepts { get; init; } = Array.Empty<MappedConceptDto>();
|
||||||
public string[] Roles { get; init; } = Array.Empty<string>();
|
public string[] Roles { get; init; } = Array.Empty<string>();
|
||||||
|
|
||||||
// Helper properties for UI compatibility
|
// Helper properties for UI compatibility
|
||||||
@@ -28,6 +30,14 @@ public record UserProfileDto
|
|||||||
public string LastReadBookTitle => LastReadBook?.Title ?? PlanConstants.DefaultActivityLabel;
|
public string LastReadBookTitle => LastReadBook?.Title ?? PlanConstants.DefaultActivityLabel;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public record MappedConceptDto
|
||||||
|
{
|
||||||
|
public string Id { get; init; } = string.Empty;
|
||||||
|
public string Type { get; init; } = string.Empty;
|
||||||
|
public string Content { get; init; } = string.Empty;
|
||||||
|
public string DisplayLabel => Content.Length > 25 ? Content.Substring(0, 22) + "..." : Content;
|
||||||
|
}
|
||||||
|
|
||||||
public record LastReadBookDto
|
public record LastReadBookDto
|
||||||
{
|
{
|
||||||
public Guid Id { get; init; }
|
public Guid Id { get; init; }
|
||||||
@@ -38,4 +48,15 @@ public record LastReadBookDto
|
|||||||
public string? LastChapter { get; init; }
|
public string? LastChapter { get; init; }
|
||||||
public int LastChapterIndex { get; init; }
|
public int LastChapterIndex { get; init; }
|
||||||
public string? Description { get; init; }
|
public string? Description { get; init; }
|
||||||
|
public bool IsReadyForReading { get; init; }
|
||||||
|
}
|
||||||
|
|
||||||
|
public record QuizResultDto
|
||||||
|
{
|
||||||
|
public Guid Id { get; init; }
|
||||||
|
public string Topic { get; init; } = string.Empty;
|
||||||
|
public int Score { get; init; }
|
||||||
|
public int TotalQuestions { get; init; }
|
||||||
|
public double Percentage { get; init; }
|
||||||
|
public DateTime CompletedDate { get; init; }
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,9 +1,27 @@
|
|||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Text.Json.Serialization;
|
||||||
|
|
||||||
namespace NexusReader.Application.Queries.Graph;
|
namespace NexusReader.Application.Queries.Graph;
|
||||||
|
|
||||||
public record GraphNodeDto(string Id, string Label, string Group, string? Type = null);
|
public record GraphNodeDto(
|
||||||
public record GraphLinkDto(string Source, string Target, string RelationType, int Value = 1);
|
[property: JsonPropertyName("id")] string Id,
|
||||||
|
[property: JsonPropertyName("label")] string Label,
|
||||||
|
[property: JsonPropertyName("group")] string Group,
|
||||||
|
[property: JsonPropertyName("description")] string? Description = null,
|
||||||
|
[property: JsonPropertyName("type")] string? Type = null,
|
||||||
|
[property: JsonPropertyName("summary")] string? Summary = null,
|
||||||
|
[property: JsonPropertyName("key_terms")] List<string>? KeyTerms = null
|
||||||
|
);
|
||||||
|
|
||||||
|
public record GraphLinkDto(
|
||||||
|
[property: JsonPropertyName("source")] string Source,
|
||||||
|
[property: JsonPropertyName("target")] string Target,
|
||||||
|
[property: JsonPropertyName("type")] string RelationType,
|
||||||
|
[property: JsonPropertyName("value")] int Value = 1
|
||||||
|
);
|
||||||
|
|
||||||
public record GraphDataDto
|
public record GraphDataDto
|
||||||
{
|
{
|
||||||
public List<GraphNodeDto> Nodes { get; init; } = new();
|
[JsonPropertyName("nodes")] public List<GraphNodeDto> Nodes { get; init; } = new();
|
||||||
public List<GraphLinkDto> Links { get; init; } = new();
|
[JsonPropertyName("links")] public List<GraphLinkDto> Links { get; init; } = new();
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -37,7 +37,8 @@ public class GetMyEbooksQueryHandler : IRequestHandler<GetMyEbooksQuery, Result<
|
|||||||
Progress = e.Progress,
|
Progress = e.Progress,
|
||||||
LastChapter = e.LastChapter ?? "Rozpoczynanie...",
|
LastChapter = e.LastChapter ?? "Rozpoczynanie...",
|
||||||
LastChapterIndex = e.LastChapterIndex,
|
LastChapterIndex = e.LastChapterIndex,
|
||||||
Description = e.Description
|
Description = e.Description,
|
||||||
|
IsReadyForReading = e.IsReadyForReading
|
||||||
})
|
})
|
||||||
.ToListAsync(cancellationToken);
|
.ToListAsync(cancellationToken);
|
||||||
|
|
||||||
|
|||||||
@@ -1,18 +1,7 @@
|
|||||||
using FluentResults;
|
using FluentResults;
|
||||||
using MediatR;
|
using MediatR;
|
||||||
using Pgvector;
|
|
||||||
using Pgvector.EntityFrameworkCore;
|
|
||||||
using NexusReader.Application.Abstractions.Services;
|
using NexusReader.Application.Abstractions.Services;
|
||||||
using NexusReader.Application.DTOs.AI;
|
using NexusReader.Application.DTOs.AI;
|
||||||
using Microsoft.Extensions.AI;
|
|
||||||
using Microsoft.EntityFrameworkCore;
|
|
||||||
using Microsoft.Extensions.Resilience;
|
|
||||||
using Polly;
|
|
||||||
using Polly.Registry;
|
|
||||||
using Mapster;
|
|
||||||
using MapsterMapper;
|
|
||||||
|
|
||||||
using NexusReader.Data.Persistence;
|
|
||||||
|
|
||||||
namespace NexusReader.Application.Queries.Library;
|
namespace NexusReader.Application.Queries.Library;
|
||||||
|
|
||||||
@@ -21,21 +10,11 @@ public record SearchLibrarySemanticallyQuery(string QueryText, string TenantId,
|
|||||||
|
|
||||||
public class SearchLibrarySemanticallyQueryHandler : IRequestHandler<SearchLibrarySemanticallyQuery, Result<List<SemanticSearchResultDto>>>
|
public class SearchLibrarySemanticallyQueryHandler : IRequestHandler<SearchLibrarySemanticallyQuery, Result<List<SemanticSearchResultDto>>>
|
||||||
{
|
{
|
||||||
private readonly IEmbeddingGenerator<string, Embedding<float>> _embeddingGenerator;
|
private readonly IKnowledgeService _knowledgeService;
|
||||||
private readonly IDbContextFactory<AppDbContext> _dbContextFactory;
|
|
||||||
private readonly ResiliencePipeline _retryPipeline;
|
|
||||||
private readonly IMapper _mapper;
|
|
||||||
|
|
||||||
public SearchLibrarySemanticallyQueryHandler(
|
public SearchLibrarySemanticallyQueryHandler(IKnowledgeService knowledgeService)
|
||||||
IEmbeddingGenerator<string, Embedding<float>> embeddingGenerator,
|
|
||||||
IDbContextFactory<AppDbContext> dbContextFactory,
|
|
||||||
ResiliencePipelineProvider<string> pipelineProvider,
|
|
||||||
IMapper mapper)
|
|
||||||
{
|
{
|
||||||
_embeddingGenerator = embeddingGenerator;
|
_knowledgeService = knowledgeService;
|
||||||
_dbContextFactory = dbContextFactory;
|
|
||||||
_retryPipeline = pipelineProvider.GetPipeline("ai-retry");
|
|
||||||
_mapper = mapper;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task<Result<List<SemanticSearchResultDto>>> Handle(SearchLibrarySemanticallyQuery request, CancellationToken cancellationToken)
|
public async Task<Result<List<SemanticSearchResultDto>>> Handle(SearchLibrarySemanticallyQuery request, CancellationToken cancellationToken)
|
||||||
@@ -45,19 +24,10 @@ public class SearchLibrarySemanticallyQueryHandler : IRequestHandler<SearchLibra
|
|||||||
return Result.Fail("Query text cannot be empty.");
|
return Result.Fail("Query text cannot be empty.");
|
||||||
}
|
}
|
||||||
|
|
||||||
// Generate embedding with retry
|
return await _knowledgeService.SearchLibrarySemanticallyAsync(
|
||||||
var embeddingResponse = await _retryPipeline.ExecuteAsync(async ct =>
|
request.QueryText,
|
||||||
await _embeddingGenerator.GenerateAsync(new[] { request.QueryText }, cancellationToken: ct), cancellationToken);
|
request.TenantId,
|
||||||
var queryVector = new Vector(embeddingResponse.First().Vector.ToArray());
|
request.Limit,
|
||||||
|
cancellationToken);
|
||||||
await using var dbContext = await _dbContextFactory.CreateDbContextAsync(cancellationToken);
|
|
||||||
var cacheEntries = await dbContext.SemanticKnowledgeCache
|
|
||||||
.Where(c => c.TenantId == request.TenantId && c.Embedding != null)
|
|
||||||
.OrderBy(c => c.Embedding!.CosineDistance(queryVector))
|
|
||||||
.Take(request.Limit)
|
|
||||||
.ToListAsync(cancellationToken);
|
|
||||||
|
|
||||||
var dtos = _mapper.Map<List<SemanticSearchResultDto>>(cacheEntries);
|
|
||||||
return Result.Ok(dtos);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -18,13 +18,15 @@ public class GetUserProfileQueryHandler : IRequestHandler<GetUserProfileQuery, R
|
|||||||
public async Task<Result<UserProfileDto>> Handle(GetUserProfileQuery request, CancellationToken cancellationToken)
|
public async Task<Result<UserProfileDto>> Handle(GetUserProfileQuery request, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
using var dbContext = await _dbContextFactory.CreateDbContextAsync(cancellationToken);
|
using var dbContext = await _dbContextFactory.CreateDbContextAsync(cancellationToken);
|
||||||
var profile = await dbContext.Users
|
|
||||||
|
var userRaw = await dbContext.Users
|
||||||
.Where(u => u.Id == request.UserId)
|
.Where(u => u.Id == request.UserId)
|
||||||
.Select(u => new UserProfileDto
|
.Select(u => new
|
||||||
{
|
{
|
||||||
Email = u.Email ?? string.Empty,
|
Email = u.Email ?? string.Empty,
|
||||||
|
UserId = u.Id,
|
||||||
AITokensUsed = u.AITokensUsed,
|
AITokensUsed = u.AITokensUsed,
|
||||||
TenantId = u.TenantId != null && u.TenantId.Length == 36 ? new Guid(u.TenantId) : Guid.Empty,
|
TenantIdString = u.TenantId,
|
||||||
Plan = u.SubscriptionPlan != null ? new SubscriptionPlanDto
|
Plan = u.SubscriptionPlan != null ? new SubscriptionPlanDto
|
||||||
{
|
{
|
||||||
Id = u.SubscriptionPlan.Id,
|
Id = u.SubscriptionPlan.Id,
|
||||||
@@ -32,9 +34,17 @@ public class GetUserProfileQueryHandler : IRequestHandler<GetUserProfileQuery, R
|
|||||||
AITokenLimit = u.SubscriptionPlan.AITokenLimit,
|
AITokenLimit = u.SubscriptionPlan.AITokenLimit,
|
||||||
MonthlyPrice = u.SubscriptionPlan.MonthlyPrice
|
MonthlyPrice = u.SubscriptionPlan.MonthlyPrice
|
||||||
} : new SubscriptionPlanDto(),
|
} : new SubscriptionPlanDto(),
|
||||||
AverageQuizScore = u.QuizResults.Any(q => q.TotalQuestions > 0)
|
QuizResults = u.QuizResults.Select(q => new
|
||||||
? (int)u.QuizResults.Where(q => q.TotalQuestions > 0).Average(q => (double)q.Score / q.TotalQuestions * 100)
|
{
|
||||||
: 0,
|
q.Score,
|
||||||
|
q.TotalQuestions,
|
||||||
|
q.Id,
|
||||||
|
q.Topic,
|
||||||
|
q.Percentage,
|
||||||
|
q.CompletedDate
|
||||||
|
}).ToList(),
|
||||||
|
DisplayName = u.DisplayName,
|
||||||
|
BooksReadCount = u.Ebooks.Count(),
|
||||||
LastReadBook = u.Ebooks.OrderByDescending(e => e.LastReadDate).Select(e => new LastReadBookDto
|
LastReadBook = u.Ebooks.OrderByDescending(e => e.LastReadDate).Select(e => new LastReadBookDto
|
||||||
{
|
{
|
||||||
Id = e.Id,
|
Id = e.Id,
|
||||||
@@ -48,7 +58,8 @@ public class GetUserProfileQueryHandler : IRequestHandler<GetUserProfileQuery, R
|
|||||||
Progress = e.Progress,
|
Progress = e.Progress,
|
||||||
LastChapter = e.LastChapter ?? "Rozpoczynanie...",
|
LastChapter = e.LastChapter ?? "Rozpoczynanie...",
|
||||||
LastChapterIndex = e.LastChapterIndex,
|
LastChapterIndex = e.LastChapterIndex,
|
||||||
Description = e.Description
|
Description = e.Description,
|
||||||
|
IsReadyForReading = e.IsReadyForReading
|
||||||
}).FirstOrDefault(),
|
}).FirstOrDefault(),
|
||||||
Roles = dbContext.UserRoles
|
Roles = dbContext.UserRoles
|
||||||
.Where(ur => ur.UserId == u.Id)
|
.Where(ur => ur.UserId == u.Id)
|
||||||
@@ -57,11 +68,59 @@ public class GetUserProfileQueryHandler : IRequestHandler<GetUserProfileQuery, R
|
|||||||
})
|
})
|
||||||
.FirstOrDefaultAsync(cancellationToken);
|
.FirstOrDefaultAsync(cancellationToken);
|
||||||
|
|
||||||
if (profile == null)
|
if (userRaw == null)
|
||||||
{
|
{
|
||||||
return Result.Fail("Profile not found.");
|
return Result.Fail("Profile not found.");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var tenantId = userRaw.TenantIdString;
|
||||||
|
var mappedConcepts = await dbContext.KnowledgeUnits
|
||||||
|
.Where(k => k.TenantId == tenantId || k.TenantId == "global" || string.IsNullOrEmpty(k.TenantId))
|
||||||
|
.OrderByDescending(k => k.CreatedAt)
|
||||||
|
.Take(6)
|
||||||
|
.Select(k => new MappedConceptDto
|
||||||
|
{
|
||||||
|
Id = k.Id,
|
||||||
|
Type = k.Type.ToString(),
|
||||||
|
Content = k.Content
|
||||||
|
})
|
||||||
|
.ToListAsync(cancellationToken);
|
||||||
|
|
||||||
|
var conceptsMappedCount = await dbContext.KnowledgeUnits
|
||||||
|
.CountAsync(k => k.TenantId == tenantId || k.TenantId == "global" || string.IsNullOrEmpty(k.TenantId), cancellationToken);
|
||||||
|
|
||||||
|
int averageQuizScore = 0;
|
||||||
|
var validQuizzes = userRaw.QuizResults.Where(q => q.TotalQuestions > 0).ToList();
|
||||||
|
if (validQuizzes.Count > 0)
|
||||||
|
{
|
||||||
|
averageQuizScore = (int)(validQuizzes.Average(q => (double)q.Score / q.TotalQuestions) * 100);
|
||||||
|
}
|
||||||
|
|
||||||
|
var profile = new UserProfileDto
|
||||||
|
{
|
||||||
|
Email = userRaw.Email,
|
||||||
|
UserId = userRaw.UserId,
|
||||||
|
AITokensUsed = userRaw.AITokensUsed,
|
||||||
|
TenantId = userRaw.TenantIdString != null && userRaw.TenantIdString.Length == 36 ? new Guid(userRaw.TenantIdString) : Guid.Empty,
|
||||||
|
Plan = userRaw.Plan,
|
||||||
|
AverageQuizScore = averageQuizScore,
|
||||||
|
DisplayName = userRaw.DisplayName,
|
||||||
|
BooksReadCount = userRaw.BooksReadCount,
|
||||||
|
ConceptsMappedCount = conceptsMappedCount,
|
||||||
|
LastReadBook = userRaw.LastReadBook,
|
||||||
|
RecentQuizzes = userRaw.QuizResults.OrderByDescending(q => q.CompletedDate).Take(5).Select(q => new QuizResultDto
|
||||||
|
{
|
||||||
|
Id = q.Id,
|
||||||
|
Topic = q.Topic,
|
||||||
|
Score = q.Score,
|
||||||
|
TotalQuestions = q.TotalQuestions,
|
||||||
|
Percentage = q.Percentage,
|
||||||
|
CompletedDate = q.CompletedDate
|
||||||
|
}).ToList(),
|
||||||
|
MappedConcepts = mappedConcepts,
|
||||||
|
Roles = userRaw.Roles
|
||||||
|
};
|
||||||
|
|
||||||
return Result.Ok(profile);
|
return Result.Ok(profile);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -55,16 +55,7 @@ public class AppDbContext : IdentityDbContext<NexusUser>
|
|||||||
entity.HasKey(e => e.ContentHash);
|
entity.HasKey(e => e.ContentHash);
|
||||||
entity.HasIndex(e => e.ContentHash).IsUnique();
|
entity.HasIndex(e => e.ContentHash).IsUnique();
|
||||||
entity.HasIndex(e => e.TenantId);
|
entity.HasIndex(e => e.TenantId);
|
||||||
if (Database.IsNpgsql())
|
entity.Ignore(e => e.Embedding);
|
||||||
{
|
|
||||||
// Configure vector column (768 dims) and HNSW index for cosine similarity
|
|
||||||
entity.Property(e => e.Embedding).HasColumnType("vector(768)");
|
|
||||||
entity.HasIndex(e => e.Embedding).HasMethod("hnsw").HasOperators("vector_cosine_ops");
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
entity.Ignore(e => e.Embedding);
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
|
|
||||||
modelBuilder.Entity<KnowledgeUnit>(entity =>
|
modelBuilder.Entity<KnowledgeUnit>(entity =>
|
||||||
|
|||||||
@@ -1,7 +1,6 @@
|
|||||||
using Microsoft.EntityFrameworkCore;
|
using Microsoft.EntityFrameworkCore;
|
||||||
using Microsoft.EntityFrameworkCore.Design;
|
using Microsoft.EntityFrameworkCore.Design;
|
||||||
using Microsoft.Extensions.Configuration;
|
using Microsoft.Extensions.Configuration;
|
||||||
using Pgvector.EntityFrameworkCore;
|
|
||||||
|
|
||||||
namespace NexusReader.Data.Persistence;
|
namespace NexusReader.Data.Persistence;
|
||||||
|
|
||||||
@@ -38,7 +37,7 @@ public class AppDbContextFactory : IDesignTimeDbContextFactory<AppDbContext>
|
|||||||
connectionString = "Host=localhost;Database=nexus_reader;Username=postgres;Password=postgres";
|
connectionString = "Host=localhost;Database=nexus_reader;Username=postgres;Password=postgres";
|
||||||
}
|
}
|
||||||
|
|
||||||
optionsBuilder.UseNpgsql(connectionString, x => x.UseVector());
|
optionsBuilder.UseNpgsql(connectionString);
|
||||||
|
|
||||||
return new AppDbContext(optionsBuilder.Options);
|
return new AppDbContext(optionsBuilder.Options);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -112,6 +112,7 @@ public static class DependencyInjection
|
|||||||
services.AddScoped<IKnowledgeService, KnowledgeService>();
|
services.AddScoped<IKnowledgeService, KnowledgeService>();
|
||||||
services.AddTransient<IEpubReader, EpubReaderService>();
|
services.AddTransient<IEpubReader, EpubReaderService>();
|
||||||
services.AddTransient<IEpubMetadataExtractor, EpubMetadataExtractor>();
|
services.AddTransient<IEpubMetadataExtractor, EpubMetadataExtractor>();
|
||||||
|
services.AddTransient<IEpubExtractor, EpubExtractor>();
|
||||||
|
|
||||||
// Fix #4: Scoped instead of Singleton — BookStorageService uses path resolution
|
// Fix #4: Scoped instead of Singleton — BookStorageService uses path resolution
|
||||||
// that is environment-specific and incompatible with Singleton lifetime in MAUI.
|
// that is environment-specific and incompatible with Singleton lifetime in MAUI.
|
||||||
@@ -119,6 +120,7 @@ public static class DependencyInjection
|
|||||||
|
|
||||||
// Fix #1: Ebook repository (scoped, matches AppDbContext lifetime)
|
// Fix #1: Ebook repository (scoped, matches AppDbContext lifetime)
|
||||||
services.AddScoped<IEbookRepository, EbookRepository>();
|
services.AddScoped<IEbookRepository, EbookRepository>();
|
||||||
|
services.AddScoped<IQuizResultRepository, QuizResultRepository>();
|
||||||
|
|
||||||
// Fix #2: SignalR broadcaster (scoped, wraps IHubContext which is itself a singleton wrapper)
|
// Fix #2: SignalR broadcaster (scoped, wraps IHubContext which is itself a singleton wrapper)
|
||||||
services.AddScoped<ISyncBroadcaster, SignalRSyncBroadcaster>();
|
services.AddScoped<ISyncBroadcaster, SignalRSyncBroadcaster>();
|
||||||
|
|||||||
@@ -46,6 +46,12 @@ internal sealed class EbookRepository : IEbookRepository
|
|||||||
_context.Ebooks.Add(ebook);
|
_context.Ebooks.Add(ebook);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <inheritdoc />
|
||||||
|
public async Task<Ebook?> FindByIdAsync(Guid id, CancellationToken cancellationToken = default)
|
||||||
|
{
|
||||||
|
return await _context.Ebooks.FindAsync(new object[] { id }, cancellationToken);
|
||||||
|
}
|
||||||
|
|
||||||
/// <inheritdoc />
|
/// <inheritdoc />
|
||||||
public Task<int> SaveChangesAsync(CancellationToken cancellationToken = default)
|
public Task<int> SaveChangesAsync(CancellationToken cancellationToken = default)
|
||||||
=> _context.SaveChangesAsync(cancellationToken);
|
=> _context.SaveChangesAsync(cancellationToken);
|
||||||
|
|||||||
@@ -0,0 +1,37 @@
|
|||||||
|
using Microsoft.EntityFrameworkCore;
|
||||||
|
using NexusReader.Application.Abstractions.Persistence;
|
||||||
|
using NexusReader.Data.Persistence;
|
||||||
|
using NexusReader.Domain.Entities;
|
||||||
|
|
||||||
|
namespace NexusReader.Infrastructure.Persistence;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// EF Core implementation of <see cref="IQuizResultRepository"/>.
|
||||||
|
/// </summary>
|
||||||
|
internal sealed class QuizResultRepository : IQuizResultRepository
|
||||||
|
{
|
||||||
|
private readonly AppDbContext _context;
|
||||||
|
|
||||||
|
public QuizResultRepository(AppDbContext context)
|
||||||
|
{
|
||||||
|
_context = context;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <inheritdoc />
|
||||||
|
public async Task<NexusUser?> FindUserByIdAsync(string userId, CancellationToken cancellationToken = default)
|
||||||
|
{
|
||||||
|
return await _context.Users.FirstOrDefaultAsync(u => u.Id == userId, cancellationToken);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <inheritdoc />
|
||||||
|
public void AddQuizResult(QuizResult quizResult)
|
||||||
|
{
|
||||||
|
_context.QuizResults.Add(quizResult);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <inheritdoc />
|
||||||
|
public Task<int> SaveChangesAsync(CancellationToken cancellationToken = default)
|
||||||
|
{
|
||||||
|
return _context.SaveChangesAsync(cancellationToken);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,85 @@
|
|||||||
|
using System.Text.RegularExpressions;
|
||||||
|
using FluentResults;
|
||||||
|
using Microsoft.Extensions.Logging;
|
||||||
|
using NexusReader.Application.Abstractions.Services;
|
||||||
|
using VersOne.Epub;
|
||||||
|
|
||||||
|
namespace NexusReader.Infrastructure.Services;
|
||||||
|
|
||||||
|
public class EpubExtractor : IEpubExtractor
|
||||||
|
{
|
||||||
|
private readonly ILogger<EpubExtractor> _logger;
|
||||||
|
|
||||||
|
public EpubExtractor(ILogger<EpubExtractor> logger)
|
||||||
|
{
|
||||||
|
_logger = logger;
|
||||||
|
}
|
||||||
|
|
||||||
|
public async Task<Result<List<string>>> ExtractChaptersTextAsync(string relativePath, CancellationToken cancellationToken = default)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var fullPath = ResolvePath(relativePath);
|
||||||
|
if (string.IsNullOrEmpty(fullPath) || !File.Exists(fullPath))
|
||||||
|
{
|
||||||
|
_logger.LogError("[EpubExtractor] EPUB file not found at path: {FilePath}", relativePath);
|
||||||
|
return Result.Fail<List<string>>($"Plik EPUB nie został znaleziony na dysku: {relativePath}");
|
||||||
|
}
|
||||||
|
|
||||||
|
using var bookRef = await EpubReader.OpenBookAsync(fullPath);
|
||||||
|
var readingOrder = bookRef.GetReadingOrder();
|
||||||
|
|
||||||
|
if (readingOrder == null || !readingOrder.Any())
|
||||||
|
{
|
||||||
|
return Result.Fail<List<string>>("EPUB nie zawiera czytelnych rozdziałów.");
|
||||||
|
}
|
||||||
|
|
||||||
|
var chapters = new List<string>();
|
||||||
|
foreach (var chapterRef in readingOrder)
|
||||||
|
{
|
||||||
|
if (cancellationToken.IsCancellationRequested)
|
||||||
|
{
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
var rawContent = await chapterRef.ReadContentAsTextAsync();
|
||||||
|
var cleanText = StripHtml(rawContent);
|
||||||
|
chapters.Add(cleanText);
|
||||||
|
}
|
||||||
|
|
||||||
|
return Result.Ok(chapters);
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
_logger.LogError(ex, "[EpubExtractor] Error extracting chapters from EPUB: {FilePath}", relativePath);
|
||||||
|
return Result.Fail<List<string>>(new Error("Failed to parse and extract text from EPUB").CausedBy(ex));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static string? ResolvePath(string relativePath)
|
||||||
|
{
|
||||||
|
var normalized = relativePath.Replace('/', Path.DirectorySeparatorChar);
|
||||||
|
var currentDir = new DirectoryInfo(AppDomain.CurrentDomain.BaseDirectory);
|
||||||
|
while (currentDir != null)
|
||||||
|
{
|
||||||
|
var candidate = Path.Combine(currentDir.FullName, "wwwroot", normalized);
|
||||||
|
if (File.Exists(candidate)) return candidate;
|
||||||
|
|
||||||
|
var devCandidate = Path.Combine(currentDir.FullName, "src", "NexusReader.Web", "wwwroot", normalized);
|
||||||
|
if (File.Exists(devCandidate)) return devCandidate;
|
||||||
|
|
||||||
|
currentDir = currentDir.Parent;
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static string StripHtml(string html)
|
||||||
|
{
|
||||||
|
if (string.IsNullOrEmpty(html)) return string.Empty;
|
||||||
|
var clean = Regex.Replace(html, @"<(style|script)\b[^>]*>.*?</\1>", "", RegexOptions.IgnoreCase | RegexOptions.Singleline);
|
||||||
|
clean = Regex.Replace(clean, @"<[^>]*>", " ");
|
||||||
|
clean = System.Net.WebUtility.HtmlDecode(clean);
|
||||||
|
clean = Regex.Replace(clean, @"\s+", " ").Trim();
|
||||||
|
return clean;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -15,6 +15,7 @@ using Polly.Registry;
|
|||||||
using Microsoft.Extensions.Options;
|
using Microsoft.Extensions.Options;
|
||||||
using NexusReader.Infrastructure.Configuration;
|
using NexusReader.Infrastructure.Configuration;
|
||||||
using Qdrant.Client;
|
using Qdrant.Client;
|
||||||
|
using Qdrant.Client.Grpc;
|
||||||
using Neo4j.Driver;
|
using Neo4j.Driver;
|
||||||
|
|
||||||
namespace NexusReader.Infrastructure.Services;
|
namespace NexusReader.Infrastructure.Services;
|
||||||
@@ -32,8 +33,9 @@ public class KnowledgeService : IKnowledgeService
|
|||||||
private readonly ILogger<KnowledgeService> _logger;
|
private readonly ILogger<KnowledgeService> _logger;
|
||||||
private readonly QdrantClient _qdrantClient;
|
private readonly QdrantClient _qdrantClient;
|
||||||
private readonly IDriver _neo4jDriver;
|
private readonly IDriver _neo4jDriver;
|
||||||
private const string PromptVersion = "1.3";
|
private const string PromptVersion = "1.7";
|
||||||
private static readonly ConcurrentDictionary<string, Lazy<Task<Result<KnowledgePacket>>>> _activeRequests = new();
|
private static readonly ConcurrentDictionary<string, Lazy<Task<Result<KnowledgePacket>>>> _activeRequests = new();
|
||||||
|
private static readonly SemaphoreSlim _collectionSemaphore = new(1, 1);
|
||||||
|
|
||||||
public KnowledgeService(
|
public KnowledgeService(
|
||||||
IChatClient chatClient,
|
IChatClient chatClient,
|
||||||
@@ -84,11 +86,12 @@ public class KnowledgeService : IKnowledgeService
|
|||||||
|
|
||||||
using var dbContext = await _dbContextFactory.CreateDbContextAsync(cancellationToken);
|
using var dbContext = await _dbContextFactory.CreateDbContextAsync(cancellationToken);
|
||||||
var normalizedText = text.Trim();
|
var normalizedText = text.Trim();
|
||||||
var hash = ContentHasher.ComputeHash(normalizedText);
|
var hashInput = $"{normalizedText}:{traceType}:{PromptVersion}";
|
||||||
|
var hash = ContentHasher.ComputeHash(hashInput);
|
||||||
|
|
||||||
// 1. Check Cache
|
// 1. Check Cache
|
||||||
var cached = await dbContext.SemanticKnowledgeCache
|
var cached = await dbContext.SemanticKnowledgeCache
|
||||||
.FirstOrDefaultAsync(c => c.ContentHash == hash && c.TenantId == tenantId, cancellationToken);
|
.FirstOrDefaultAsync(c => c.ContentHash == hash, cancellationToken);
|
||||||
|
|
||||||
if (cached != null && cached.PromptVersion == PromptVersion)
|
if (cached != null && cached.PromptVersion == PromptVersion)
|
||||||
{
|
{
|
||||||
@@ -96,7 +99,12 @@ public class KnowledgeService : IKnowledgeService
|
|||||||
try
|
try
|
||||||
{
|
{
|
||||||
var packet = JsonSerializer.Deserialize<KnowledgePacket>(cached.JsonData, JsonOptions);
|
var packet = JsonSerializer.Deserialize<KnowledgePacket>(cached.JsonData, JsonOptions);
|
||||||
if (packet != null) return Result.Ok(packet);
|
if (packet != null)
|
||||||
|
{
|
||||||
|
await ProcessKnowledgeUnitsAsync(packet, tenantId, ebookId, dbContext, cancellationToken);
|
||||||
|
await dbContext.SaveChangesAsync(cancellationToken);
|
||||||
|
return Result.Ok(packet);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
catch (JsonException ex)
|
catch (JsonException ex)
|
||||||
{
|
{
|
||||||
@@ -105,7 +113,7 @@ public class KnowledgeService : IKnowledgeService
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Deduplicate concurrent active requests for the exact same hash
|
// Deduplicate concurrent active requests for the exact same hash
|
||||||
var requestKey = $"{tenantId}:{hash}:{traceType}";
|
var requestKey = $"{hash}:{traceType}";
|
||||||
|
|
||||||
var lazyTask = _activeRequests.GetOrAdd(requestKey, k =>
|
var lazyTask = _activeRequests.GetOrAdd(requestKey, k =>
|
||||||
new Lazy<Task<Result<KnowledgePacket>>>(
|
new Lazy<Task<Result<KnowledgePacket>>>(
|
||||||
@@ -177,7 +185,7 @@ public class KnowledgeService : IKnowledgeService
|
|||||||
|
|
||||||
// 4. Save to Cache
|
// 4. Save to Cache
|
||||||
var cached = await dbContext.SemanticKnowledgeCache
|
var cached = await dbContext.SemanticKnowledgeCache
|
||||||
.FirstOrDefaultAsync(c => c.ContentHash == hash && c.TenantId == tenantId);
|
.FirstOrDefaultAsync(c => c.ContentHash == hash);
|
||||||
|
|
||||||
var cacheEntry = new SemanticKnowledgeCache
|
var cacheEntry = new SemanticKnowledgeCache
|
||||||
{
|
{
|
||||||
@@ -201,7 +209,14 @@ public class KnowledgeService : IKnowledgeService
|
|||||||
// 5. Process structured KnowledgeUnits (Graph Expansion)
|
// 5. Process structured KnowledgeUnits (Graph Expansion)
|
||||||
await ProcessKnowledgeUnitsAsync(knowledgePacket, tenantId, ebookId, dbContext, default);
|
await ProcessKnowledgeUnitsAsync(knowledgePacket, tenantId, ebookId, dbContext, default);
|
||||||
|
|
||||||
await dbContext.SaveChangesAsync();
|
try
|
||||||
|
{
|
||||||
|
await dbContext.SaveChangesAsync();
|
||||||
|
}
|
||||||
|
catch (DbUpdateException ex) when (ex.InnerException is Npgsql.PostgresException pgEx && pgEx.SqlState == "23505")
|
||||||
|
{
|
||||||
|
_logger.LogWarning("[KnowledgeService] Concurrency collision on SemanticKnowledgeCache for {Hash}; another process saved it first. Swallowing.", hash);
|
||||||
|
}
|
||||||
return Result.Ok(knowledgePacket);
|
return Result.Ok(knowledgePacket);
|
||||||
}
|
}
|
||||||
catch (JsonException ex)
|
catch (JsonException ex)
|
||||||
@@ -224,6 +239,30 @@ public class KnowledgeService : IKnowledgeService
|
|||||||
|
|
||||||
private async Task ProcessKnowledgeUnitsAsync(KnowledgePacket packet, string tenantId, Guid? ebookId, AppDbContext dbContext, CancellationToken cancellationToken)
|
private async Task ProcessKnowledgeUnitsAsync(KnowledgePacket packet, string tenantId, Guid? ebookId, AppDbContext dbContext, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
|
if (packet.Graph != null && (packet.Units == null || !packet.Units.Any()))
|
||||||
|
{
|
||||||
|
var graphUnits = packet.Graph.Nodes.Select(node => new KnowledgeUnitDto(
|
||||||
|
node.Id,
|
||||||
|
node.Type ?? "concept",
|
||||||
|
node.Description ?? node.Label,
|
||||||
|
new Dictionary<string, object>
|
||||||
|
{
|
||||||
|
["label"] = node.Label,
|
||||||
|
["group"] = node.Group,
|
||||||
|
["summary"] = node.Summary ?? "",
|
||||||
|
["key_terms"] = node.KeyTerms ?? new List<string>()
|
||||||
|
}
|
||||||
|
)).ToList();
|
||||||
|
|
||||||
|
var graphLinks = packet.Graph.Links.Select(link => new KnowledgeLinkDto(
|
||||||
|
link.Source,
|
||||||
|
link.Target,
|
||||||
|
link.RelationType
|
||||||
|
)).ToList();
|
||||||
|
|
||||||
|
packet = packet with { Units = graphUnits, Links = graphLinks };
|
||||||
|
}
|
||||||
|
|
||||||
var unitIds = packet.Units.Select(u => u.Id).ToList();
|
var unitIds = packet.Units.Select(u => u.Id).ToList();
|
||||||
var linkSourceIds = packet.Links.Select(l => l.Source).ToList();
|
var linkSourceIds = packet.Links.Select(l => l.Source).ToList();
|
||||||
var linkTargetIds = packet.Links.Select(l => l.Target).ToList();
|
var linkTargetIds = packet.Links.Select(l => l.Target).ToList();
|
||||||
@@ -285,6 +324,192 @@ public class KnowledgeService : IKnowledgeService
|
|||||||
_logger.LogWarning("[KnowledgeService] Skipping invalid link {Source} -> {Target}: one or both units are missing.", linkDto.Source, linkDto.Target);
|
_logger.LogWarning("[KnowledgeService] Skipping invalid link {Source} -> {Target}: one or both units are missing.", linkDto.Source, linkDto.Target);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Generate and upsert vectors to Qdrant in batch
|
||||||
|
var unitsToEmbed = packet.Units
|
||||||
|
.Where(u => !string.IsNullOrEmpty(u.Content))
|
||||||
|
.ToList();
|
||||||
|
|
||||||
|
if (unitsToEmbed.Any())
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var contents = unitsToEmbed.Select(u => u.Content).ToList();
|
||||||
|
|
||||||
|
var embeddingResponse = await _retryPipeline.ExecuteAsync(async ct =>
|
||||||
|
await _embeddingGenerator.GenerateAsync(
|
||||||
|
contents,
|
||||||
|
new EmbeddingGenerationOptions { Dimensions = 768 },
|
||||||
|
cancellationToken: ct), cancellationToken);
|
||||||
|
|
||||||
|
var embeddings = embeddingResponse.ToList();
|
||||||
|
var points = new List<PointStruct>();
|
||||||
|
|
||||||
|
for (int i = 0; i < unitsToEmbed.Count; i++)
|
||||||
|
{
|
||||||
|
var unitDto = unitsToEmbed[i];
|
||||||
|
var vector = embeddings[i].Vector.ToArray();
|
||||||
|
|
||||||
|
var point = new PointStruct
|
||||||
|
{
|
||||||
|
Id = GetDeterministicGuid(unitDto.Id),
|
||||||
|
Vectors = vector,
|
||||||
|
Payload =
|
||||||
|
{
|
||||||
|
["content"] = unitDto.Content,
|
||||||
|
["type"] = unitDto.Type ?? string.Empty,
|
||||||
|
["tenantId"] = tenantId,
|
||||||
|
["ebookId"] = ebookId?.ToString() ?? string.Empty,
|
||||||
|
["metadataJson"] = JsonSerializer.Serialize(unitDto.Metadata)
|
||||||
|
}
|
||||||
|
};
|
||||||
|
points.Add(point);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (points.Any())
|
||||||
|
{
|
||||||
|
await EnsureCollectionExistsAsync("knowledge_units", cancellationToken);
|
||||||
|
await _qdrantClient.UpsertAsync("knowledge_units", points, cancellationToken: cancellationToken);
|
||||||
|
_logger.LogInformation("[KnowledgeService] Successfully upserted {Count} points to Qdrant collection 'knowledge_units'.", points.Count);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
_logger.LogError(ex, "[KnowledgeService] Failed to generate and upsert embeddings for knowledge units to Qdrant.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 6. Synchronize to Neo4j graph database
|
||||||
|
await SyncToNeo4jAsync(packet, cancellationToken);
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task SyncToNeo4jAsync(KnowledgePacket packet, CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
if (packet.Units == null || !packet.Units.Any()) return;
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
await using var session = _neo4jDriver.AsyncSession();
|
||||||
|
|
||||||
|
// 1. Merge nodes in a transaction
|
||||||
|
await session.ExecuteWriteAsync(async tx =>
|
||||||
|
{
|
||||||
|
foreach (var unit in packet.Units)
|
||||||
|
{
|
||||||
|
var cypher = @"
|
||||||
|
MERGE (u:KnowledgeUnit {id: $id})
|
||||||
|
ON CREATE SET u.content = $content, u.type = $type
|
||||||
|
ON MATCH SET u.content = $content, u.type = $type";
|
||||||
|
|
||||||
|
var guidStr = GetDeterministicGuid(unit.Id).ToString();
|
||||||
|
await tx.RunAsync(cypher, new
|
||||||
|
{
|
||||||
|
id = guidStr,
|
||||||
|
content = unit.Content ?? string.Empty,
|
||||||
|
type = unit.Type ?? "concept"
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 2. Merge links in a transaction
|
||||||
|
if (packet.Links != null && packet.Links.Any())
|
||||||
|
{
|
||||||
|
await session.ExecuteWriteAsync(async tx =>
|
||||||
|
{
|
||||||
|
foreach (var link in packet.Links)
|
||||||
|
{
|
||||||
|
if (string.IsNullOrWhiteSpace(link.Source) || string.IsNullOrWhiteSpace(link.Target))
|
||||||
|
continue;
|
||||||
|
|
||||||
|
var relationType = string.IsNullOrWhiteSpace(link.Relation) ? "RELATED_TO" : link.Relation.Trim().ToUpperInvariant();
|
||||||
|
relationType = System.Text.RegularExpressions.Regex.Replace(relationType, @"[^A-Z0-9_]", "_");
|
||||||
|
if (string.IsNullOrEmpty(relationType) || relationType == "_")
|
||||||
|
{
|
||||||
|
relationType = "RELATED_TO";
|
||||||
|
}
|
||||||
|
|
||||||
|
var cypher = $@"
|
||||||
|
MATCH (source:KnowledgeUnit {{id: $sourceId}})
|
||||||
|
MATCH (target:KnowledgeUnit {{id: $targetId}})
|
||||||
|
MERGE (source)-[r:{relationType}]->(target)";
|
||||||
|
|
||||||
|
var sourceGuidStr = GetDeterministicGuid(link.Source).ToString();
|
||||||
|
var targetGuidStr = GetDeterministicGuid(link.Target).ToString();
|
||||||
|
|
||||||
|
await tx.RunAsync(cypher, new
|
||||||
|
{
|
||||||
|
sourceId = sourceGuidStr,
|
||||||
|
targetId = targetGuidStr
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
_logger.LogInformation("[KnowledgeService] Successfully synchronized {NodeCount} nodes and {LinkCount} links to Neo4j.", packet.Units.Count, packet.Links?.Count ?? 0);
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
_logger.LogError(ex, "[KnowledgeService] Failed to synchronize knowledge graph to Neo4j.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task EnsureCollectionExistsAsync(string collectionName, CancellationToken cancellationToken = default)
|
||||||
|
{
|
||||||
|
await _collectionSemaphore.WaitAsync(cancellationToken);
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var exists = await _qdrantClient.CollectionExistsAsync(collectionName, cancellationToken);
|
||||||
|
if (!exists)
|
||||||
|
{
|
||||||
|
_logger.LogInformation("[KnowledgeService] Creating Qdrant collection '{CollectionName}'...", collectionName);
|
||||||
|
await _qdrantClient.CreateCollectionAsync(
|
||||||
|
collectionName: collectionName,
|
||||||
|
vectorsConfig: new VectorParams
|
||||||
|
{
|
||||||
|
Size = 768,
|
||||||
|
Distance = Distance.Cosine
|
||||||
|
},
|
||||||
|
cancellationToken: cancellationToken
|
||||||
|
);
|
||||||
|
_logger.LogInformation("[KnowledgeService] Qdrant collection '{CollectionName}' created successfully.", collectionName);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
if (ex.Message.Contains("already exists", StringComparison.OrdinalIgnoreCase) ||
|
||||||
|
(ex.InnerException != null && ex.InnerException.Message.Contains("already exists", StringComparison.OrdinalIgnoreCase)))
|
||||||
|
{
|
||||||
|
_logger.LogInformation("[KnowledgeService] Qdrant collection '{CollectionName}' was already created by another thread.", collectionName);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
_logger.LogError(ex, "[KnowledgeService] Error ensuring Qdrant collection '{CollectionName}' exists.", collectionName);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
finally
|
||||||
|
{
|
||||||
|
_collectionSemaphore.Release();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static Guid GetDeterministicGuid(string input)
|
||||||
|
{
|
||||||
|
if (Guid.TryParse(input, out var guid))
|
||||||
|
{
|
||||||
|
return guid;
|
||||||
|
}
|
||||||
|
|
||||||
|
using var md5 = System.Security.Cryptography.MD5.Create();
|
||||||
|
byte[] hash = md5.ComputeHash(System.Text.Encoding.UTF8.GetBytes(input));
|
||||||
|
return new Guid(hash);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static string GetPointIdString(PointId pointId)
|
||||||
|
{
|
||||||
|
if (pointId == null) return string.Empty;
|
||||||
|
return pointId.PointIdOptionsCase == PointId.PointIdOptionsOneofCase.Uuid
|
||||||
|
? pointId.Uuid
|
||||||
|
: pointId.Num.ToString();
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task<Result<GroundednessResult>> VerifyGroundednessAsync(string answer, string context, string tenantId, CancellationToken cancellationToken = default)
|
public async Task<Result<GroundednessResult>> VerifyGroundednessAsync(string answer, string context, string tenantId, CancellationToken cancellationToken = default)
|
||||||
@@ -354,6 +579,7 @@ public class KnowledgeService : IKnowledgeService
|
|||||||
List<Qdrant.Client.Grpc.ScoredPoint> searchResult;
|
List<Qdrant.Client.Grpc.ScoredPoint> searchResult;
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
|
await EnsureCollectionExistsAsync("knowledge_units", cancellationToken);
|
||||||
var response = await _qdrantClient.SearchAsync(
|
var response = await _qdrantClient.SearchAsync(
|
||||||
collectionName: "knowledge_units",
|
collectionName: "knowledge_units",
|
||||||
vector: queryVector,
|
vector: queryVector,
|
||||||
@@ -363,15 +589,37 @@ public class KnowledgeService : IKnowledgeService
|
|||||||
);
|
);
|
||||||
searchResult = response.ToList();
|
searchResult = response.ToList();
|
||||||
}
|
}
|
||||||
catch (Exception)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
|
_logger.LogWarning(ex, "[KnowledgeService] Qdrant search failed during GetRelevantContextAsync. Returning empty search results.");
|
||||||
searchResult = new List<Qdrant.Client.Grpc.ScoredPoint>();
|
searchResult = new List<Qdrant.Client.Grpc.ScoredPoint>();
|
||||||
}
|
}
|
||||||
|
|
||||||
var contexts = searchResult.Select(point => new RelevantContext
|
var contexts = searchResult.Select(point =>
|
||||||
{
|
{
|
||||||
Text = point.Payload.TryGetValue("content", out var cv) ? cv.StringValue : string.Empty,
|
var content = point.Payload.TryGetValue("content", out var cv) ? cv.StringValue : string.Empty;
|
||||||
Confidence = point.Score
|
var summary = string.Empty;
|
||||||
|
if (point.Payload.TryGetValue("metadataJson", out var metaVal) && !string.IsNullOrEmpty(metaVal.StringValue))
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var meta = JsonSerializer.Deserialize<Dictionary<string, object>>(metaVal.StringValue);
|
||||||
|
if (meta != null && meta.TryGetValue("summary", out var sumObj))
|
||||||
|
{
|
||||||
|
summary = sumObj?.ToString();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (JsonException ex)
|
||||||
|
{
|
||||||
|
_logger.LogWarning(ex, "[KnowledgeService] Failed to deserialize metadata JSON in RelevantContext mapping.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
var text = string.IsNullOrEmpty(summary) ? content : $"{content}: {summary}";
|
||||||
|
return new RelevantContext
|
||||||
|
{
|
||||||
|
Text = text,
|
||||||
|
Confidence = point.Score
|
||||||
|
};
|
||||||
}).ToList();
|
}).ToList();
|
||||||
|
|
||||||
return Result.Ok(contexts);
|
return Result.Ok(contexts);
|
||||||
@@ -417,6 +665,7 @@ public class KnowledgeService : IKnowledgeService
|
|||||||
List<Qdrant.Client.Grpc.ScoredPoint> searchResult;
|
List<Qdrant.Client.Grpc.ScoredPoint> searchResult;
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
|
await EnsureCollectionExistsAsync("knowledge_units", cancellationToken);
|
||||||
var response = await _qdrantClient.SearchAsync(
|
var response = await _qdrantClient.SearchAsync(
|
||||||
collectionName: "knowledge_units",
|
collectionName: "knowledge_units",
|
||||||
vector: queryVector,
|
vector: queryVector,
|
||||||
@@ -438,7 +687,7 @@ public class KnowledgeService : IKnowledgeService
|
|||||||
}
|
}
|
||||||
|
|
||||||
// 3. Graph Expansion via Neo4j
|
// 3. Graph Expansion via Neo4j
|
||||||
var candidateIds = searchResult.Select(r => r.Id.ToString()).ToList();
|
var candidateIds = searchResult.Select(r => GetPointIdString(r.Id)).ToList();
|
||||||
var definitions = new Dictionary<string, List<string>>();
|
var definitions = new Dictionary<string, List<string>>();
|
||||||
|
|
||||||
if (candidateIds.Any())
|
if (candidateIds.Any())
|
||||||
@@ -447,7 +696,7 @@ public class KnowledgeService : IKnowledgeService
|
|||||||
{
|
{
|
||||||
await using var session = _neo4jDriver.AsyncSession();
|
await using var session = _neo4jDriver.AsyncSession();
|
||||||
var cypher = @"
|
var cypher = @"
|
||||||
MATCH (source:KnowledgeUnit)-[r:DEFINES]->(target:KnowledgeUnit)
|
MATCH (source:KnowledgeUnit)-[r]->(target:KnowledgeUnit)
|
||||||
WHERE source.id IN $candidateIds
|
WHERE source.id IN $candidateIds
|
||||||
RETURN source.id AS sourceId, target.content AS targetContent";
|
RETURN source.id AS sourceId, target.content AS targetContent";
|
||||||
|
|
||||||
@@ -516,12 +765,15 @@ public class KnowledgeService : IKnowledgeService
|
|||||||
{
|
{
|
||||||
metadata = JsonSerializer.Deserialize<Dictionary<string, object>>(metaVal.StringValue);
|
metadata = JsonSerializer.Deserialize<Dictionary<string, object>>(metaVal.StringValue);
|
||||||
}
|
}
|
||||||
catch {}
|
catch (JsonException ex)
|
||||||
|
{
|
||||||
|
_logger.LogWarning(ex, "[KnowledgeService] Failed to deserialize metadata JSON in search library mapping.");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
var dto = new SemanticSearchResultDto
|
var dto = new SemanticSearchResultDto
|
||||||
{
|
{
|
||||||
ContentHash = point.Id.ToString(),
|
ContentHash = GetPointIdString(point.Id),
|
||||||
Snippet = content,
|
Snippet = content,
|
||||||
UnitType = type,
|
UnitType = type,
|
||||||
RelevanceScore = point.Score,
|
RelevanceScore = point.Score,
|
||||||
@@ -529,7 +781,7 @@ public class KnowledgeService : IKnowledgeService
|
|||||||
Metadata = metadata
|
Metadata = metadata
|
||||||
};
|
};
|
||||||
|
|
||||||
var pointIdStr = point.Id.ToString();
|
var pointIdStr = GetPointIdString(point.Id);
|
||||||
if (definitions.TryGetValue(pointIdStr, out var pointDefs) && pointDefs.Any())
|
if (definitions.TryGetValue(pointIdStr, out var pointDefs) && pointDefs.Any())
|
||||||
{
|
{
|
||||||
dto.Snippet = $"[Context: {string.Join("; ", pointDefs)}]\n{dto.Snippet}";
|
dto.Snippet = $"[Context: {string.Join("; ", pointDefs)}]\n{dto.Snippet}";
|
||||||
@@ -602,6 +854,7 @@ public class KnowledgeService : IKnowledgeService
|
|||||||
List<Qdrant.Client.Grpc.ScoredPoint> searchResult;
|
List<Qdrant.Client.Grpc.ScoredPoint> searchResult;
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
|
await EnsureCollectionExistsAsync("knowledge_units", cancellationToken);
|
||||||
var response = await _qdrantClient.SearchAsync(
|
var response = await _qdrantClient.SearchAsync(
|
||||||
collectionName: "knowledge_units",
|
collectionName: "knowledge_units",
|
||||||
vector: queryVector,
|
vector: queryVector,
|
||||||
@@ -627,11 +880,28 @@ public class KnowledgeService : IKnowledgeService
|
|||||||
}
|
}
|
||||||
|
|
||||||
// 3. Graph Expansion via Neo4j
|
// 3. Graph Expansion via Neo4j
|
||||||
var candidateIds = searchResult.Select(r => r.Id.ToString()).ToList();
|
var candidateIds = searchResult.Select(r => GetPointIdString(r.Id)).ToList();
|
||||||
var relatedContexts = new List<string>();
|
var relatedContexts = new List<string>();
|
||||||
|
|
||||||
// Keep map of point ID -> payload data for fast mapping later
|
// Keep map of point ID -> payload data for fast mapping later
|
||||||
var pointMap = searchResult.ToDictionary(r => r.Id.ToString(), r => r);
|
var pointMap = searchResult.ToDictionary(r => GetPointIdString(r.Id), r => r);
|
||||||
|
|
||||||
|
// Fetch knowledge units from PostgreSQL to map Guids back to rich metadata summaries
|
||||||
|
var guidMap = new Dictionary<string, KnowledgeUnit>();
|
||||||
|
try
|
||||||
|
{
|
||||||
|
using var dbContext = await _dbContextFactory.CreateDbContextAsync(cancellationToken);
|
||||||
|
var units = await dbContext.KnowledgeUnits
|
||||||
|
.Include(u => u.Ebook)
|
||||||
|
.ThenInclude(e => e.Author)
|
||||||
|
.Where(u => u.TenantId == tenantId && (ebookId == null || u.EbookId == ebookId))
|
||||||
|
.ToListAsync(cancellationToken);
|
||||||
|
guidMap = units.ToDictionary(u => GetDeterministicGuid(u.Id).ToString(), u => u);
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
_logger.LogWarning(ex, "[KnowledgeService] Failed to load KnowledgeUnits from PostgreSQL for Guid mapping.");
|
||||||
|
}
|
||||||
|
|
||||||
if (candidateIds.Any())
|
if (candidateIds.Any())
|
||||||
{
|
{
|
||||||
@@ -641,7 +911,7 @@ public class KnowledgeService : IKnowledgeService
|
|||||||
var cypher = @"
|
var cypher = @"
|
||||||
MATCH (source:KnowledgeUnit)
|
MATCH (source:KnowledgeUnit)
|
||||||
WHERE source.id IN $candidateIds
|
WHERE source.id IN $candidateIds
|
||||||
OPTIONAL MATCH (source)-[r:DEFINES|RELATED_TO]->(target:KnowledgeUnit)
|
OPTIONAL MATCH (source)-[r]->(target:KnowledgeUnit)
|
||||||
RETURN source.id AS sourceId, source.content AS sourceContent,
|
RETURN source.id AS sourceId, source.content AS sourceContent,
|
||||||
collect({ targetId: target.id, targetContent: target.content, relation: type(r) }) AS relations";
|
collect({ targetId: target.id, targetContent: target.content, relation: type(r) }) AS relations";
|
||||||
|
|
||||||
@@ -654,23 +924,70 @@ public class KnowledgeService : IKnowledgeService
|
|||||||
foreach (var record in neoResult)
|
foreach (var record in neoResult)
|
||||||
{
|
{
|
||||||
var sourceId = record["sourceId"].As<string>();
|
var sourceId = record["sourceId"].As<string>();
|
||||||
var sourceContent = record["sourceContent"].As<string>();
|
|
||||||
|
|
||||||
relatedContexts.Add($"[Source ID: {sourceId}] {sourceContent}");
|
var sourceText = string.Empty;
|
||||||
|
if (guidMap.TryGetValue(sourceId, out var sourceUnit))
|
||||||
|
{
|
||||||
|
var summary = string.Empty;
|
||||||
|
if (!string.IsNullOrEmpty(sourceUnit.MetadataJson))
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var meta = JsonSerializer.Deserialize<Dictionary<string, object>>(sourceUnit.MetadataJson);
|
||||||
|
if (meta != null && meta.TryGetValue("summary", out var sumObj))
|
||||||
|
{
|
||||||
|
summary = sumObj?.ToString();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (JsonException jsonEx)
|
||||||
|
{
|
||||||
|
_logger.LogWarning(jsonEx, "[KnowledgeService] Failed to deserialize metadata JSON for unit {UnitId} in AskQuestionAsync source hydration.", sourceUnit.Id);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
sourceText = string.IsNullOrEmpty(summary) ? sourceUnit.Content : $"{sourceUnit.Content}: {summary}";
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
sourceText = record["sourceContent"].As<string>();
|
||||||
|
}
|
||||||
|
|
||||||
|
relatedContexts.Add($"[Source ID: {sourceId}] {sourceText}");
|
||||||
|
|
||||||
var relations = record["relations"].As<List<object>>();
|
var relations = record["relations"].As<List<object>>();
|
||||||
if (relations != null)
|
if (relations != null)
|
||||||
{
|
{
|
||||||
foreach (var relObj in relations)
|
foreach (var relObj in relations)
|
||||||
{
|
{
|
||||||
if (relObj is Dictionary<string, object> relDict &&
|
if (relObj is System.Collections.IDictionary relDict)
|
||||||
relDict.TryGetValue("targetId", out var targetIdVal) && targetIdVal is string targetId &&
|
|
||||||
relDict.TryGetValue("targetContent", out var targetContentVal) && targetContentVal is string targetContent &&
|
|
||||||
relDict.TryGetValue("relation", out var relationVal) && relationVal is string relation)
|
|
||||||
{
|
{
|
||||||
if (!string.IsNullOrEmpty(targetContent))
|
var targetId = relDict["targetId"]?.ToString();
|
||||||
|
var targetContent = relDict["targetContent"]?.ToString();
|
||||||
|
var relation = relDict["relation"]?.ToString();
|
||||||
|
|
||||||
|
if (!string.IsNullOrEmpty(targetContent) && !string.IsNullOrEmpty(relation))
|
||||||
{
|
{
|
||||||
relatedContexts.Add($"[Related Context ({relation}) to {sourceId}] {targetContent}");
|
var targetText = targetContent;
|
||||||
|
if (!string.IsNullOrEmpty(targetId) && guidMap.TryGetValue(targetId, out var targetUnit))
|
||||||
|
{
|
||||||
|
var summary = string.Empty;
|
||||||
|
if (!string.IsNullOrEmpty(targetUnit.MetadataJson))
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var meta = JsonSerializer.Deserialize<Dictionary<string, object>>(targetUnit.MetadataJson);
|
||||||
|
if (meta != null && meta.TryGetValue("summary", out var sumObj))
|
||||||
|
{
|
||||||
|
summary = sumObj?.ToString();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (JsonException jsonEx)
|
||||||
|
{
|
||||||
|
_logger.LogWarning(jsonEx, "[KnowledgeService] Failed to deserialize metadata JSON for unit {UnitId} in AskQuestionAsync target hydration.", targetUnit.Id);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
targetText = string.IsNullOrEmpty(summary) ? targetUnit.Content : $"{targetUnit.Content}: {summary}";
|
||||||
|
}
|
||||||
|
relatedContexts.Add($"[Related Context ({relation}) to {sourceId}] {targetText}");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -682,9 +999,35 @@ public class KnowledgeService : IKnowledgeService
|
|||||||
_logger.LogWarning(ex, "[KnowledgeService] Neo4j graph expansion failed. Falling back to direct Qdrant points.");
|
_logger.LogWarning(ex, "[KnowledgeService] Neo4j graph expansion failed. Falling back to direct Qdrant points.");
|
||||||
foreach (var point in searchResult)
|
foreach (var point in searchResult)
|
||||||
{
|
{
|
||||||
var sourceId = point.Id.ToString();
|
var sourceId = GetPointIdString(point.Id);
|
||||||
var content = point.Payload.TryGetValue("content", out var cv) ? cv.StringValue : string.Empty;
|
|
||||||
relatedContexts.Add($"[Source ID: {sourceId}] {content}");
|
var sourceText = string.Empty;
|
||||||
|
if (guidMap.TryGetValue(sourceId, out var sourceUnit))
|
||||||
|
{
|
||||||
|
var summary = string.Empty;
|
||||||
|
if (!string.IsNullOrEmpty(sourceUnit.MetadataJson))
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var meta = JsonSerializer.Deserialize<Dictionary<string, object>>(sourceUnit.MetadataJson);
|
||||||
|
if (meta != null && meta.TryGetValue("summary", out var sumObj))
|
||||||
|
{
|
||||||
|
summary = sumObj?.ToString();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (JsonException jsonEx)
|
||||||
|
{
|
||||||
|
_logger.LogWarning(jsonEx, "[KnowledgeService] Failed to deserialize metadata JSON for unit {UnitId} in fallback AskQuestionAsync.", sourceUnit.Id);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
sourceText = string.IsNullOrEmpty(summary) ? sourceUnit.Content : $"{sourceUnit.Content}: {summary}";
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
sourceText = point.Payload.TryGetValue("content", out var cv) ? cv.StringValue : string.Empty;
|
||||||
|
}
|
||||||
|
|
||||||
|
relatedContexts.Add($"[Source ID: {sourceId}] {sourceText}");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -708,33 +1051,14 @@ public class KnowledgeService : IKnowledgeService
|
|||||||
// 5. Build prompt and invoke Gemini with structured JSON formatting
|
// 5. Build prompt and invoke Gemini with structured JSON formatting
|
||||||
var contextBlocksText = string.Join("\n\n", relatedContexts);
|
var contextBlocksText = string.Join("\n\n", relatedContexts);
|
||||||
|
|
||||||
var systemPrompt = @"
|
var systemPrompt = PromptRegistry.GroundedRAGSystemPrompt;
|
||||||
You are an advanced, extremely precise Fact-Checking AI assistant. Your task is to answer the user's question using ONLY the provided context blocks.
|
|
||||||
|
|
||||||
Strict Grounding Rules:
|
|
||||||
1. Rely EXCLUSIVELY on the provided context. Do NOT use any pre-existing external knowledge, facts, or assumptions.
|
|
||||||
2. If the context does not contain the answer, you must state exactly: 'I cannot answer this based on the provided book context.'
|
|
||||||
3. For every statement or claim you make in your answer, you must cite the specific source IDs (e.g., source chunk ID or hash) from the context.
|
|
||||||
4. You must format your response ONLY as a JSON object matching the following structure:
|
|
||||||
{
|
|
||||||
""answer"": ""The answer text goes here, referencing [Source ID] as citations."",
|
|
||||||
""citations"": [
|
|
||||||
{
|
|
||||||
""citationId"": ""The exact source ID cited (e.g., chunk hash/ID)"",
|
|
||||||
""snippet"": ""The precise sentence or phrase from the context that supports this statement."",
|
|
||||||
""sourceBook"": ""The book title or 'Unknown'""
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
";
|
|
||||||
|
|
||||||
var userPrompt = $"Context:\n{contextBlocksText}\n\nQuestion: {question}";
|
var userPrompt = $"Context:\n{contextBlocksText}\n\nQuestion: {question}";
|
||||||
|
|
||||||
var options = new ChatOptions
|
var options = new ChatOptions
|
||||||
{
|
{
|
||||||
Temperature = 0.0f,
|
Temperature = 0.0f,
|
||||||
MaxOutputTokens = 1500,
|
MaxOutputTokens = 1500
|
||||||
ResponseFormat = ChatResponseFormat.Json
|
|
||||||
};
|
};
|
||||||
|
|
||||||
var chatResponse = await _retryPipeline.ExecuteAsync(async ct =>
|
var chatResponse = await _retryPipeline.ExecuteAsync(async ct =>
|
||||||
@@ -746,6 +1070,20 @@ Strict Grounding Rules:
|
|||||||
|
|
||||||
var rawJson = chatResponse.Text?.Trim() ?? string.Empty;
|
var rawJson = chatResponse.Text?.Trim() ?? string.Empty;
|
||||||
rawJson = rawJson.Replace("```json", "").Replace("```", "").Trim();
|
rawJson = rawJson.Replace("```json", "").Replace("```", "").Trim();
|
||||||
|
|
||||||
|
// Handle direct text fallback when model bypasses JSON format
|
||||||
|
if (!rawJson.StartsWith("{") &&
|
||||||
|
(rawJson.Contains("cannot answer", StringComparison.OrdinalIgnoreCase) ||
|
||||||
|
rawJson.Contains("context does not contain", StringComparison.OrdinalIgnoreCase) ||
|
||||||
|
rawJson.Contains("provided book context", StringComparison.OrdinalIgnoreCase)))
|
||||||
|
{
|
||||||
|
return Result.Ok(new GroundedResponseDto
|
||||||
|
{
|
||||||
|
Answer = "I cannot answer this based on the provided book context.",
|
||||||
|
Citations = new List<CitationDto>()
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
rawJson = JsonRepairHelper.Repair(rawJson);
|
rawJson = JsonRepairHelper.Repair(rawJson);
|
||||||
|
|
||||||
try
|
try
|
||||||
@@ -756,15 +1094,42 @@ Strict Grounding Rules:
|
|||||||
return Result.Fail("Failed to deserialize grounded RAG response.");
|
return Result.Fail("Failed to deserialize grounded RAG response.");
|
||||||
}
|
}
|
||||||
|
|
||||||
// Hydrate book titles for citations if unknown
|
// Hydrate book titles, author, and page number for citations if unknown
|
||||||
foreach (var citation in groundedResult.Citations)
|
foreach (var citation in groundedResult.Citations)
|
||||||
{
|
{
|
||||||
if (pointMap.TryGetValue(citation.CitationId, out var point) &&
|
if (pointMap.TryGetValue(citation.CitationId, out var point) &&
|
||||||
point.Payload.TryGetValue("ebookId", out var ev) &&
|
point.Payload.TryGetValue("ebookId", out var ev) &&
|
||||||
Guid.TryParse(ev.StringValue, out var ebId) &&
|
Guid.TryParse(ev.StringValue, out var ebId))
|
||||||
ebookTitles.TryGetValue(ebId, out var title))
|
|
||||||
{
|
{
|
||||||
citation.SourceBook = title;
|
if (ebookTitles.TryGetValue(ebId, out var title))
|
||||||
|
{
|
||||||
|
citation.SourceBook = title;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Look up from guidMap to get exact page number and author
|
||||||
|
if (guidMap.TryGetValue(citation.CitationId, out var unit))
|
||||||
|
{
|
||||||
|
if (unit.Ebook?.Author != null)
|
||||||
|
{
|
||||||
|
citation.Author = unit.Ebook.Author.Name;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!string.IsNullOrEmpty(unit.MetadataJson))
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var meta = JsonSerializer.Deserialize<Dictionary<string, object>>(unit.MetadataJson);
|
||||||
|
if (meta != null && meta.TryGetValue("page", out var pageObj) && int.TryParse(pageObj?.ToString(), out var pageVal))
|
||||||
|
{
|
||||||
|
citation.PageNumber = pageVal;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (JsonException ex)
|
||||||
|
{
|
||||||
|
_logger.LogWarning(ex, "[KnowledgeService] Failed to deserialize metadata JSON for unit {UnitId} in AskQuestionAsync citation mapping.", unit.Id);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -790,6 +1155,30 @@ Strict Grounding Rules:
|
|||||||
await dbContext.SemanticKnowledgeCache.ExecuteDeleteAsync(cancellationToken);
|
await dbContext.SemanticKnowledgeCache.ExecuteDeleteAsync(cancellationToken);
|
||||||
await dbContext.KnowledgeUnits.ExecuteDeleteAsync(cancellationToken);
|
await dbContext.KnowledgeUnits.ExecuteDeleteAsync(cancellationToken);
|
||||||
await dbContext.KnowledgeUnitLinks.ExecuteDeleteAsync(cancellationToken);
|
await dbContext.KnowledgeUnitLinks.ExecuteDeleteAsync(cancellationToken);
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
await _qdrantClient.DeleteCollectionAsync("knowledge_units", cancellationToken: cancellationToken);
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
_logger.LogWarning(ex, "[KnowledgeService] Failed to drop Qdrant collection 'knowledge_units' during cache clear.");
|
||||||
|
}
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
await using var session = _neo4jDriver.AsyncSession();
|
||||||
|
await session.ExecuteWriteAsync(async tx =>
|
||||||
|
{
|
||||||
|
await tx.RunAsync("MATCH (n:KnowledgeUnit) DETACH DELETE n");
|
||||||
|
});
|
||||||
|
_logger.LogInformation("[KnowledgeService] Successfully wiped Neo4j 'KnowledgeUnit' nodes.");
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
_logger.LogWarning(ex, "[KnowledgeService] Failed to wipe Neo4j graph during cache clear.");
|
||||||
|
}
|
||||||
|
|
||||||
return Result.Ok();
|
return Result.Ok();
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
|
|||||||
@@ -4,9 +4,10 @@ public static class PromptRegistry
|
|||||||
{
|
{
|
||||||
public const string KnowledgeExtractionSystemPrompt =
|
public const string KnowledgeExtractionSystemPrompt =
|
||||||
"You are an expert educator. Analyze the provided text to extract key concepts, generate relevant quizzes, and construct a knowledge graph. " +
|
"You are an expert educator. Analyze the provided text to extract key concepts, generate relevant quizzes, and construct a knowledge graph. " +
|
||||||
"CRITICAL: Restrict 'concept.label' to a maximum of 3 words (e.g., 'Dependency Injection' instead of full sentences). " +
|
"**LANGUAGE CRITICAL**: Detect the language of the provided text. You MUST generate all human-readable fields ('title', 'description', 'question', 'options', 'label') in the EXACT SAME LANGUAGE as the source text. Do NOT translate them to English unless the source text is in English. " +
|
||||||
|
"CRITICAL: Restrict 'concept.label' to a maximum of 3 words (e.g., 'Dependency Injection' or its exact foreign equivalent, never full sentences). " +
|
||||||
"CRITICAL: Extract a MAXIMUM of 15 key concepts/plot points from the text. " +
|
"CRITICAL: Extract a MAXIMUM of 15 key concepts/plot points from the text. " +
|
||||||
"CRITICAL: Code blocks (e.g., markdown code snippets) must be excluded from the relationship graph, or summarized as a single node (e.g., 'Code Example'). Do NOT create nodes for variables, functions, namespaces, or individual lines of code. " +
|
"CRITICAL: Code blocks (e.g., markdown code snippets) must be excluded from the relationship graph, or summarized as a single node with the label 'Code Example' translated to the detected language. Do NOT create nodes for variables, functions, namespaces, or individual lines of code. " +
|
||||||
"CRITICAL: Return ONLY a minified JSON object. Do NOT include markdown formatting like ```json or ```. Do NOT include explanations. " +
|
"CRITICAL: Return ONLY a minified JSON object. Do NOT include markdown formatting like ```json or ```. Do NOT include explanations. " +
|
||||||
"Schema: { " +
|
"Schema: { " +
|
||||||
"\"concepts\": [ { \"title\": \"string\", \"description\": \"string\" } ], " +
|
"\"concepts\": [ { \"title\": \"string\", \"description\": \"string\" } ], " +
|
||||||
@@ -15,28 +16,66 @@ public static class PromptRegistry
|
|||||||
"}.";
|
"}.";
|
||||||
|
|
||||||
public const string GraphExtractionPrompt =
|
public const string GraphExtractionPrompt =
|
||||||
"You are an expert at information architecture. Extract key concepts and paragraph mappings from the text to build a unified knowledge graph. " +
|
"You are a strict Minimalist Information Architect. Your sole job is to build a high-level, sparse linear backbone for a textbook chapter. " +
|
||||||
"The input text consists of several paragraphs, each starting with its unique block ID in the format '[ID: seg-X]'. " +
|
"**LANGUAGE CRITICAL**: Detect the language of the provided text. The 'label', 'summary', and 'key_terms' fields MUST be in the EXACT SAME LANGUAGE as the source text. " +
|
||||||
"Extract two types of nodes: " +
|
"The input text consists of sections starting with block IDs (e.g., '[ID: seg-4]'). " +
|
||||||
"1. Concept Nodes (group: 'concept'): Extract the main technical concepts discussed (e.g., ID: 'dependency-injection', label: 'Dependency Injection'). Max 10 concepts. Labels must be at most 3 words. " +
|
"CRITICAL TOPOLOGY RULES (ZERO TOLERANCE FOR CLUTTER): " +
|
||||||
"2. Block Nodes (group: 'current'): For each paragraph in the input, create a node representing that paragraph where 'id' is the exact block ID (e.g., 'seg-1'), and 'label' is a brief summary of that paragraph's content (max 3 words). " +
|
"1. HARD NODE LIMIT: You are strictly forbidden from extracting more than 4 to 5 nodes IN TOTAL for the entire text. If there are more sections, select ONLY the 4-5 absolute most critical, high-level structural pillars. " +
|
||||||
"CRITICAL: If a paragraph is a code block, represent it as a single block node with label 'Code Example' (group: 'current'). Do NOT extract low-level code elements (like variables, classes, methods, or namespaces) as separate concept nodes. " +
|
"2. NO CONCEPT CLOUDS: Do NOT create nodes for individual technologies, files, terms, or phrases (e.g., 'Kestrel', 'appsettings.json', 'DI', 'Blazor Server' must NEVER be nodes). They must ONLY exist as text strings inside the 'key_terms' array of a major node. " +
|
||||||
"CRITICAL: Connect related concept nodes together, and connect each concept node to the block nodes ('seg-X') where it is discussed. " +
|
"3. LINEAR SPINE PATTERN: Nodes must form a clear, clean path or simple tree representing the chronological reading journey (e.g., Node 1 -> Node 2 -> Node 3). Do NOT create complex web loops or interconnect every node. Limit total links in the entire JSON to maximum 4 or 5 links. " +
|
||||||
"Limit connections to a MAXIMUM of 15 most relevant links. " +
|
"4. NODE DATA STRUCTURE: " +
|
||||||
"Return ONLY minified JSON. Schema: { \"graph\": { \"nodes\": [ { \"id\": \"string\", \"label\": \"string\", \"group\": \"concept|current\" } ], \"links\": [ { \"source\": \"string\", \"target\": \"string\", \"value\": 1 } ] } }";
|
" - 'id': must be the exact block ID (e.g., 'seg-16'). " +
|
||||||
|
" - 'label': clear technical title (Max 3 words, e.g., 'Blazor Hosting Models'). " +
|
||||||
|
" - 'group': strictly either 'bridge' (if it compares legacy vs modern) or 'concept' (for standalone core pillars). " +
|
||||||
|
" - 'summary': exact 2-sentence distillation for the Contextual Panel. " +
|
||||||
|
" - 'key_terms': array of max 5 short strings representing the micro-concepts hidden inside this section. " +
|
||||||
|
"System keys configuration: All JSON keys ('nodes', 'links', 'id', 'label', 'group', 'summary', 'key_terms', 'source', 'target', 'type') must remain strictly in English. " +
|
||||||
|
"Return ONLY minified JSON. Schema: " +
|
||||||
|
"{ " +
|
||||||
|
" \"graph\": { " +
|
||||||
|
" \"nodes\": [ " +
|
||||||
|
" { \"id\": \"seg-X\", \"label\": \"string\", \"group\": \"concept|bridge\", \"summary\": \"string\", \"key_terms\": [ \"string\" ] } " +
|
||||||
|
" ], " +
|
||||||
|
" \"links\": [ " +
|
||||||
|
" { \"source\": \"seg-X\", \"target\": \"seg-Y\", \"type\": \"maps_to|contains\" } " +
|
||||||
|
" ] " +
|
||||||
|
" } " +
|
||||||
|
"}";
|
||||||
|
|
||||||
public const string SummaryAndQuizPrompt =
|
public const string SummaryAndQuizPrompt =
|
||||||
"You are an expert educator. Provide a concise summary of the text and generate a challenging quiz (3-5 questions). " +
|
"You are an expert educator. Provide a concise summary of the text and generate a challenging quiz (3-5 questions). " +
|
||||||
|
"**LANGUAGE CRITICAL**: Detect the language of the provided text. The generated 'summary', 'question', and 'options' MUST be in the EXACT SAME LANGUAGE as the source text. " +
|
||||||
"Return ONLY minified JSON. Schema: { \"summary\": \"string\", \"quizzes\": [ { \"question\": \"string\", \"options\": [ \"string\" ], \"correct_index\": 0 } ] }";
|
"Return ONLY minified JSON. Schema: { \"summary\": \"string\", \"quizzes\": [ { \"question\": \"string\", \"options\": [ \"string\" ], \"correct_index\": 0 } ] }";
|
||||||
|
|
||||||
public const string KM_ExtractionPrompt =
|
public const string KM_ExtractionPrompt =
|
||||||
"You are an expert at Knowledge Engineering. Segment the provided text into discrete Knowledge Units. " +
|
"You are an expert at Knowledge Engineering. Segment the provided text into discrete Knowledge Units. " +
|
||||||
|
"**LANGUAGE CRITICAL**: Detect the language of the provided text. The 'content' field MUST be in the EXACT SAME LANGUAGE as the source text. " +
|
||||||
"Identify 'units' (sections, tables, definitions, rules) and 'links' (how they relate). " +
|
"Identify 'units' (sections, tables, definitions, rules) and 'links' (how they relate). " +
|
||||||
"CRITICAL: Units must be granular. " +
|
"CRITICAL: Units must be granular. " +
|
||||||
"CRITICAL: Code blocks must be summarized under the parent unit or represented as a single 'Code Example' unit. Do NOT segment code blocks into granular low-level code details (e.g., classes, variables, parameters). " +
|
"CRITICAL: Code blocks must be summarized under the parent unit or represented as a single 'Code Example' unit (translate the name to the detected language). Do NOT segment code blocks into granular low-level code details (e.g., classes, variables, parameters). " +
|
||||||
|
"CRITICAL SYSTEM VALUES: The fields 'type' (strictly: 'Section', 'Table', 'Definition', or 'Rule') and 'relation' (strictly: 'Next', 'Defines', 'Contains', or 'References') are system keys and MUST remain in English as specified. " +
|
||||||
"Schema: { " +
|
"Schema: { " +
|
||||||
"\"units\": [ { \"id\": \"string\", \"type\": \"Section|Table|Definition|Rule\", \"content\": \"string\", \"metadata\": { \"page\": 0 } } ], " +
|
"\"units\": [ { \"id\": \"string\", \"type\": \"Section|Table|Definition|Rule\", \"content\": \"string\", \"metadata\": { \"page\": 0 } } ], " +
|
||||||
"\"links\": [ { \"source\": \"string\", \"target\": \"string\", \"relation\": \"Next|Defines|Contains|References\" } ] " +
|
"\"links\": [ { \"source\": \"string\", \"target\": \"string\", \"relation\": \"Next|Defines|Contains|References\" } ] " +
|
||||||
"}.";
|
"}.";
|
||||||
|
|
||||||
|
public const string GroundedRAGSystemPrompt = """
|
||||||
|
You are an advanced, extremely precise Fact-Checking AI assistant. Your task is to answer the user's question using ONLY the provided context blocks.
|
||||||
|
|
||||||
|
Strict Grounding Rules:
|
||||||
|
1. Rely EXCLUSIVELY on the provided context. Do NOT use any pre-existing external knowledge, facts, or assumptions.
|
||||||
|
2. If the context does not contain the answer, you must set the "answer" property in the JSON object exactly to: 'I cannot answer this based on the provided book context.' and the "citations" array must be empty.
|
||||||
|
3. For every statement or claim you make in your answer, you must cite the specific source IDs (e.g., source chunk ID or hash) from the context.
|
||||||
|
4. You must format your response ONLY as a JSON object matching the following structure:
|
||||||
|
{
|
||||||
|
"answer": "The answer text goes here, referencing [Source ID] as citations.",
|
||||||
|
"citations": [
|
||||||
|
{
|
||||||
|
"citationId": "The exact source ID cited (e.g., chunk hash/ID)",
|
||||||
|
"snippet": "The precise sentence or phrase from the context that supports this statement.",
|
||||||
|
"sourceBook": "The book title or 'Unknown'"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
""";
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -8,4 +8,22 @@ public partial class App : Microsoft.Maui.Controls.Application
|
|||||||
|
|
||||||
MainPage = new MainPage();
|
MainPage = new MainPage();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected override Window CreateWindow(IActivationState? activationState)
|
||||||
|
{
|
||||||
|
var window = base.CreateWindow(activationState);
|
||||||
|
|
||||||
|
// Hook into native MAUI lifecycle events to cleanly flush and close Serilog buffers
|
||||||
|
window.Stopped += (s, e) =>
|
||||||
|
{
|
||||||
|
Serilog.Log.CloseAndFlush();
|
||||||
|
};
|
||||||
|
|
||||||
|
window.Destroying += (s, e) =>
|
||||||
|
{
|
||||||
|
Serilog.Log.CloseAndFlush();
|
||||||
|
};
|
||||||
|
|
||||||
|
return window;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,144 @@
|
|||||||
|
using System.Net.Http.Headers;
|
||||||
|
using System.Threading;
|
||||||
|
using Microsoft.Extensions.Configuration;
|
||||||
|
using Microsoft.Extensions.DependencyInjection;
|
||||||
|
using NexusReader.Application.Abstractions.Services;
|
||||||
|
|
||||||
|
namespace NexusReader.Maui.Infrastructure.Identity;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// A secure HTTP message delegating handler for MAUI that automatically appends JWT tokens
|
||||||
|
/// to trusted origin requests and transparently refreshes expired tokens in a thread-safe manner.
|
||||||
|
/// </summary>
|
||||||
|
public class MobileAuthenticationHeaderHandler : DelegatingHandler
|
||||||
|
{
|
||||||
|
private readonly INativeStorageService _storageService;
|
||||||
|
private readonly IServiceProvider _serviceProvider;
|
||||||
|
private const string TokenKey = "nexus_auth_token";
|
||||||
|
private static readonly SemaphoreSlim _refreshSemaphore = new(1, 1);
|
||||||
|
|
||||||
|
public MobileAuthenticationHeaderHandler(INativeStorageService storageService, IServiceProvider serviceProvider)
|
||||||
|
{
|
||||||
|
_storageService = storageService;
|
||||||
|
_serviceProvider = serviceProvider;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override async Task<HttpResponseMessage> SendAsync(HttpRequestMessage request, CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
var path = request.RequestUri?.AbsolutePath ?? "";
|
||||||
|
bool isAuthEndpoint = path.Contains("identity/login") ||
|
||||||
|
path.Contains("identity/register") ||
|
||||||
|
path.Contains("identity/refresh");
|
||||||
|
|
||||||
|
// Resolve configured API host dynamically to avoid hardcoded IP addresses
|
||||||
|
var config = _serviceProvider.GetRequiredService<IConfiguration>();
|
||||||
|
var apiBaseUrlString = config["ApiSettings:BaseUrl"];
|
||||||
|
string? apiHost = null;
|
||||||
|
if (!string.IsNullOrEmpty(apiBaseUrlString) && Uri.TryCreate(apiBaseUrlString, UriKind.Absolute, out var apiUri))
|
||||||
|
{
|
||||||
|
apiHost = apiUri.Host;
|
||||||
|
}
|
||||||
|
|
||||||
|
// In MAUI, since we only call our own local or staging APIs, we trust local IP/localhost/configured API host.
|
||||||
|
// We ensure we don't accidentally leak tokens to third-party endpoints.
|
||||||
|
bool isTrustedHost = request.RequestUri != null &&
|
||||||
|
(request.RequestUri.Host == "localhost" ||
|
||||||
|
request.RequestUri.Host == "127.0.0.1" ||
|
||||||
|
(apiHost != null && request.RequestUri.Host == apiHost) ||
|
||||||
|
request.RequestUri.Host.EndsWith("nexusreader.com")); // Or staging domains
|
||||||
|
|
||||||
|
string? originalToken = null;
|
||||||
|
|
||||||
|
if (!isAuthEndpoint && isTrustedHost)
|
||||||
|
{
|
||||||
|
var tokenResult = await _storageService.GetSecureString(TokenKey);
|
||||||
|
if (tokenResult.IsSuccess && !string.IsNullOrEmpty(tokenResult.Value))
|
||||||
|
{
|
||||||
|
originalToken = tokenResult.Value;
|
||||||
|
request.Headers.Authorization = new AuthenticationHeaderValue("Bearer", originalToken);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var response = await base.SendAsync(request, cancellationToken);
|
||||||
|
|
||||||
|
// Transparent JWT Auto-Refresh on 401 Unauthorized
|
||||||
|
if (response.StatusCode == System.Net.HttpStatusCode.Unauthorized && !isAuthEndpoint)
|
||||||
|
{
|
||||||
|
await _refreshSemaphore.WaitAsync(cancellationToken);
|
||||||
|
try
|
||||||
|
{
|
||||||
|
// Re-read token to verify if another concurrent request already refreshed it
|
||||||
|
var tokenResult = await _storageService.GetSecureString(TokenKey);
|
||||||
|
var currentToken = tokenResult.IsSuccess ? tokenResult.Value : null;
|
||||||
|
|
||||||
|
bool refreshed = false;
|
||||||
|
|
||||||
|
if (!string.IsNullOrEmpty(currentToken) && currentToken != originalToken)
|
||||||
|
{
|
||||||
|
refreshed = true;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
using var scope = _serviceProvider.CreateScope();
|
||||||
|
var identityService = scope.ServiceProvider.GetRequiredService<IIdentityService>();
|
||||||
|
var refreshResult = await identityService.RefreshTokenAsync();
|
||||||
|
if (refreshResult.IsSuccess)
|
||||||
|
{
|
||||||
|
var newTokenResult = await _storageService.GetSecureString(TokenKey);
|
||||||
|
currentToken = newTokenResult.IsSuccess ? newTokenResult.Value : null;
|
||||||
|
refreshed = !string.IsNullOrEmpty(currentToken);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
await identityService.LogoutAsync();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (refreshed && !string.IsNullOrEmpty(currentToken))
|
||||||
|
{
|
||||||
|
var newRequest = await CloneHttpRequestMessageAsync(request);
|
||||||
|
newRequest.Headers.Authorization = new AuthenticationHeaderValue("Bearer", currentToken);
|
||||||
|
return await base.SendAsync(newRequest, cancellationToken);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
Serilog.Log.Error(ex, "[MobileAuthHandler] Automated token renewal failed");
|
||||||
|
}
|
||||||
|
finally
|
||||||
|
{
|
||||||
|
_refreshSemaphore.Release();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return response;
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task<HttpRequestMessage> CloneHttpRequestMessageAsync(HttpRequestMessage req)
|
||||||
|
{
|
||||||
|
var clone = new HttpRequestMessage(req.Method, req.RequestUri)
|
||||||
|
{
|
||||||
|
Version = req.Version
|
||||||
|
};
|
||||||
|
|
||||||
|
if (req.Content != null)
|
||||||
|
{
|
||||||
|
var ms = new System.IO.MemoryStream();
|
||||||
|
await req.Content.CopyToAsync(ms);
|
||||||
|
ms.Position = 0;
|
||||||
|
clone.Content = new StreamContent(ms);
|
||||||
|
|
||||||
|
foreach (var h in req.Content.Headers)
|
||||||
|
{
|
||||||
|
clone.Content.Headers.TryAddWithoutValidation(h.Key, h.Value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach (var h in req.Headers)
|
||||||
|
{
|
||||||
|
clone.Headers.TryAddWithoutValidation(h.Key, h.Value);
|
||||||
|
}
|
||||||
|
|
||||||
|
return clone;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,53 @@
|
|||||||
|
using Microsoft.Extensions.Logging;
|
||||||
|
using Microsoft.JSInterop;
|
||||||
|
|
||||||
|
namespace NexusReader.Maui.Infrastructure.Logging;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Lightweight bridge service that intercepts logs, console outputs, and uncaught exceptions
|
||||||
|
/// from the Blazor WebView/JS side, and routes them directly to Serilog under "BlazorWebView" context.
|
||||||
|
/// </summary>
|
||||||
|
public sealed class BlazorLoggingBridge
|
||||||
|
{
|
||||||
|
private readonly ILogger _logger;
|
||||||
|
|
||||||
|
public BlazorLoggingBridge(ILoggerFactory loggerFactory)
|
||||||
|
{
|
||||||
|
_logger = loggerFactory.CreateLogger("BlazorWebView");
|
||||||
|
}
|
||||||
|
|
||||||
|
[JSInvokable("LogJsMessage")]
|
||||||
|
public void LogJsMessage(string level, string message, string? stackTrace = null)
|
||||||
|
{
|
||||||
|
if (string.IsNullOrWhiteSpace(message))
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
switch (level.ToLowerInvariant())
|
||||||
|
{
|
||||||
|
case "error":
|
||||||
|
case "exception":
|
||||||
|
if (!string.IsNullOrWhiteSpace(stackTrace))
|
||||||
|
{
|
||||||
|
_logger.LogError("JS Unhandled Exception: {Message}\nStack Trace:\n{StackTrace}", message, stackTrace);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
_logger.LogError("JS Error: {Message}", message);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case "warning":
|
||||||
|
case "warn":
|
||||||
|
_logger.LogWarning("JS Warning: {Message}", message);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case "info":
|
||||||
|
case "log":
|
||||||
|
default:
|
||||||
|
_logger.LogInformation("JS Log: {Message}", message);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,106 @@
|
|||||||
|
using Microsoft.Extensions.Logging;
|
||||||
|
using Serilog;
|
||||||
|
using Serilog.Core;
|
||||||
|
using Serilog.Events;
|
||||||
|
using Serilog.Formatting;
|
||||||
|
using Serilog.Formatting.Display;
|
||||||
|
|
||||||
|
namespace NexusReader.Maui.Infrastructure.Logging;
|
||||||
|
|
||||||
|
public static class SerilogConfiguration
|
||||||
|
{
|
||||||
|
private const string OutputTemplate =
|
||||||
|
"[{Timestamp:yyyy-MM-dd HH:mm:ss.fff zzz}] [{Level:u3}] [ThreadId: {ThreadId}] [{SourceContext}] {Message:lj}{NewLine}{Exception}";
|
||||||
|
|
||||||
|
public static MauiAppBuilder RegisterLogging(this MauiAppBuilder builder)
|
||||||
|
{
|
||||||
|
// 1. Ensure logs directory exists in secure sandbox
|
||||||
|
var logDir = Path.Combine(Microsoft.Maui.Storage.FileSystem.AppDataDirectory, "logs");
|
||||||
|
if (!Directory.Exists(logDir))
|
||||||
|
{
|
||||||
|
Directory.CreateDirectory(logDir);
|
||||||
|
}
|
||||||
|
var logPath = Path.Combine(logDir, "log-.txt");
|
||||||
|
|
||||||
|
// 2. Inject sandboxed log path dynamically into configuration provider
|
||||||
|
builder.Configuration["Serilog:WriteTo:0:Args:configure:0:Args:path"] = logPath;
|
||||||
|
|
||||||
|
// 3. Configure Serilog Logger Configuration using App Configuration settings
|
||||||
|
var loggerConfig = new LoggerConfiguration()
|
||||||
|
.ReadFrom.Configuration(builder.Configuration)
|
||||||
|
.Enrich.With(new ThreadIdEnricher());
|
||||||
|
|
||||||
|
// 4. Platform-specific and environment-specific sinks
|
||||||
|
#if ANDROID
|
||||||
|
// Direct Native Android Logcat Sink (JNI bindings for native diagnostics)
|
||||||
|
loggerConfig.WriteTo.Sink(
|
||||||
|
new AndroidLogcatSink(new MessageTemplateTextFormatter(OutputTemplate, null)),
|
||||||
|
restrictedToMinimumLevel: LogEventLevel.Debug);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
// 5. Initialize the static Serilog Log
|
||||||
|
Log.Logger = loggerConfig.CreateLogger();
|
||||||
|
|
||||||
|
// 6. Connect Serilog to Microsoft.Extensions.Logging
|
||||||
|
builder.Logging.ClearProviders();
|
||||||
|
builder.Logging.AddSerilog(dispose: true);
|
||||||
|
|
||||||
|
return builder;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// A custom self-contained thread enricher to avoid unnecessary NuGet packages.
|
||||||
|
/// </summary>
|
||||||
|
internal sealed class ThreadIdEnricher : ILogEventEnricher
|
||||||
|
{
|
||||||
|
public void Enrich(LogEvent logEvent, ILogEventPropertyFactory propertyFactory)
|
||||||
|
{
|
||||||
|
logEvent.AddPropertyIfAbsent(propertyFactory.CreateProperty("ThreadId", Environment.CurrentManagedThreadId));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#if ANDROID
|
||||||
|
/// <summary>
|
||||||
|
/// A high-performance, direct Android Logcat Sink utilizing native Android APIs.
|
||||||
|
/// </summary>
|
||||||
|
internal sealed class AndroidLogcatSink : ILogEventSink
|
||||||
|
{
|
||||||
|
private readonly ITextFormatter _formatter;
|
||||||
|
private const string Tag = "NexusReader";
|
||||||
|
|
||||||
|
public AndroidLogcatSink(ITextFormatter formatter)
|
||||||
|
{
|
||||||
|
_formatter = formatter ?? throw new ArgumentNullException(nameof(formatter));
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Emit(LogEvent logEvent)
|
||||||
|
{
|
||||||
|
using var writer = new StringWriter();
|
||||||
|
_formatter.Format(logEvent, writer);
|
||||||
|
var message = writer.ToString().Trim();
|
||||||
|
|
||||||
|
switch (logEvent.Level)
|
||||||
|
{
|
||||||
|
case LogEventLevel.Verbose:
|
||||||
|
Android.Util.Log.Verbose(Tag, message);
|
||||||
|
break;
|
||||||
|
case LogEventLevel.Debug:
|
||||||
|
Android.Util.Log.Debug(Tag, message);
|
||||||
|
break;
|
||||||
|
case LogEventLevel.Information:
|
||||||
|
Android.Util.Log.Info(Tag, message);
|
||||||
|
break;
|
||||||
|
case LogEventLevel.Warning:
|
||||||
|
Android.Util.Log.Warn(Tag, message);
|
||||||
|
break;
|
||||||
|
case LogEventLevel.Error:
|
||||||
|
Android.Util.Log.Error(Tag, message);
|
||||||
|
break;
|
||||||
|
case LogEventLevel.Fatal:
|
||||||
|
Android.Util.Log.Wtf(Tag, message);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif
|
||||||
@@ -1,5 +1,8 @@
|
|||||||
@using Microsoft.AspNetCore.Components.Routing
|
@using Microsoft.AspNetCore.Components.Routing
|
||||||
@using NexusReader.UI.Shared
|
@using NexusReader.UI.Shared
|
||||||
|
@using NexusReader.Maui.Infrastructure.Logging
|
||||||
|
@inject IJSRuntime JSRuntime
|
||||||
|
@inject BlazorLoggingBridge LoggingBridge
|
||||||
|
|
||||||
<Router AppAssembly="@typeof(NexusReader.UI.Shared._Imports).Assembly">
|
<Router AppAssembly="@typeof(NexusReader.UI.Shared._Imports).Assembly">
|
||||||
<Found Context="routeData">
|
<Found Context="routeData">
|
||||||
@@ -16,3 +19,21 @@
|
|||||||
</LayoutView>
|
</LayoutView>
|
||||||
</NotFound>
|
</NotFound>
|
||||||
</Router>
|
</Router>
|
||||||
|
|
||||||
|
@code {
|
||||||
|
protected override async Task OnAfterRenderAsync(bool firstRender)
|
||||||
|
{
|
||||||
|
if (firstRender)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var dotNetRef = DotNetObjectReference.Create(LoggingBridge);
|
||||||
|
await JSRuntime.InvokeVoidAsync("NexusLogging.initializeBridge", dotNetRef);
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
System.Diagnostics.Debug.WriteLine($"[SerilogBridge] Failed to initialize Blazor/JS Bridge: {ex}");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,9 +1,13 @@
|
|||||||
using Microsoft.Extensions.Logging;
|
using Microsoft.Extensions.Logging;
|
||||||
|
using Microsoft.Extensions.Configuration;
|
||||||
|
using Microsoft.Extensions.DependencyInjection;
|
||||||
using NexusReader.Application.Abstractions.Services;
|
using NexusReader.Application.Abstractions.Services;
|
||||||
using NexusReader.Infrastructure.Mobile.Services;
|
using NexusReader.Infrastructure.Mobile.Services;
|
||||||
using NexusReader.UI.Shared.Services;
|
using NexusReader.UI.Shared.Services;
|
||||||
using NexusReader.Application;
|
using NexusReader.Application;
|
||||||
using MediatR;
|
using MediatR;
|
||||||
|
using NexusReader.Maui.Infrastructure.Logging;
|
||||||
|
using NexusReader.Maui.Infrastructure.Identity;
|
||||||
|
|
||||||
namespace NexusReader.Maui;
|
namespace NexusReader.Maui;
|
||||||
|
|
||||||
@@ -14,16 +18,30 @@ public static class MauiProgram
|
|||||||
try
|
try
|
||||||
{
|
{
|
||||||
var builder = MauiApp.CreateBuilder();
|
var builder = MauiApp.CreateBuilder();
|
||||||
|
|
||||||
|
// Load embedded appsettings.json configuration
|
||||||
|
var assembly = typeof(App).Assembly;
|
||||||
|
using (var stream = assembly.GetManifestResourceStream("NexusReader.Maui.appsettings.json"))
|
||||||
|
{
|
||||||
|
if (stream != null)
|
||||||
|
{
|
||||||
|
((IConfigurationBuilder)builder.Configuration).AddJsonStream(stream);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
builder
|
builder
|
||||||
.UseMauiApp<App>();
|
.UseMauiApp<App>()
|
||||||
|
.RegisterLogging();
|
||||||
|
|
||||||
builder.Services.AddMauiBlazorWebView();
|
builder.Services.AddMauiBlazorWebView();
|
||||||
|
|
||||||
#if DEBUG
|
#if DEBUG
|
||||||
builder.Services.AddBlazorWebViewDeveloperTools();
|
builder.Services.AddBlazorWebViewDeveloperTools();
|
||||||
builder.Logging.AddDebug();
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
// Interception bridge for JS/Blazor WebView logs
|
||||||
|
builder.Services.AddSingleton<BlazorLoggingBridge>();
|
||||||
|
|
||||||
// Minimal Infrastructure
|
// Minimal Infrastructure
|
||||||
builder.Services.AddSingleton<IPlatformService, MauiPlatformService>();
|
builder.Services.AddSingleton<IPlatformService, MauiPlatformService>();
|
||||||
builder.Services.AddSingleton<INativeStorageService, MauiStorageService>();
|
builder.Services.AddSingleton<INativeStorageService, MauiStorageService>();
|
||||||
@@ -34,8 +52,15 @@ public static class MauiProgram
|
|||||||
sp.GetRequiredService<NexusAuthenticationStateProvider>());
|
sp.GetRequiredService<NexusAuthenticationStateProvider>());
|
||||||
builder.Services.AddAuthorizationCore();
|
builder.Services.AddAuthorizationCore();
|
||||||
|
|
||||||
// Basic Network
|
// Basic Network with Secure Token Handler
|
||||||
builder.Services.AddScoped(sp => new HttpClient { BaseAddress = new Uri("http://10.0.2.2:5000") });
|
builder.Services.AddTransient<MobileAuthenticationHeaderHandler>();
|
||||||
|
builder.Services.AddHttpClient("NexusAPI", client =>
|
||||||
|
{
|
||||||
|
var apiBaseUrl = builder.Configuration["ApiSettings:BaseUrl"] ?? "http://localhost:5000";
|
||||||
|
client.BaseAddress = new Uri(apiBaseUrl);
|
||||||
|
}).AddHttpMessageHandler<MobileAuthenticationHeaderHandler>();
|
||||||
|
|
||||||
|
builder.Services.AddScoped(sp => sp.GetRequiredService<IHttpClientFactory>().CreateClient("NexusAPI"));
|
||||||
|
|
||||||
// UI State
|
// UI State
|
||||||
builder.Services.AddScoped<IThemeService, ThemeService>();
|
builder.Services.AddScoped<IThemeService, ThemeService>();
|
||||||
|
|||||||
@@ -27,6 +27,14 @@
|
|||||||
<PackageReference Include="Microsoft.Extensions.Logging.Debug" Version="10.0.0" />
|
<PackageReference Include="Microsoft.Extensions.Logging.Debug" Version="10.0.0" />
|
||||||
<PackageReference Include="Microsoft.AspNetCore.Components.WebView.Maui" Version="10.0.20" />
|
<PackageReference Include="Microsoft.AspNetCore.Components.WebView.Maui" Version="10.0.20" />
|
||||||
<PackageReference Include="Microsoft.Maui.Essentials" Version="10.0.20" />
|
<PackageReference Include="Microsoft.Maui.Essentials" Version="10.0.20" />
|
||||||
|
<PackageReference Include="Serilog" Version="4.1.0" />
|
||||||
|
<PackageReference Include="Serilog.Extensions.Logging" Version="8.0.0" />
|
||||||
|
<PackageReference Include="Serilog.Sinks.File" Version="6.0.0" />
|
||||||
|
<PackageReference Include="Serilog.Sinks.Async" Version="2.1.0" />
|
||||||
|
<PackageReference Include="Serilog.Sinks.Debug" Version="3.0.0" />
|
||||||
|
<PackageReference Include="Serilog.Settings.Configuration" Version="8.0.4" />
|
||||||
|
<PackageReference Include="Microsoft.Extensions.Configuration.Json" Version="10.0.0" />
|
||||||
|
<PackageReference Include="Microsoft.Extensions.Http" Version="10.0.0" />
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
@@ -34,4 +42,8 @@
|
|||||||
<ProjectReference Include="..\NexusReader.Infrastructure.Mobile\NexusReader.Infrastructure.Mobile.csproj" />
|
<ProjectReference Include="..\NexusReader.Infrastructure.Mobile\NexusReader.Infrastructure.Mobile.csproj" />
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|
||||||
|
<ItemGroup>
|
||||||
|
<EmbeddedResource Include="appsettings.json" />
|
||||||
|
</ItemGroup>
|
||||||
|
|
||||||
</Project>
|
</Project>
|
||||||
|
|||||||
@@ -0,0 +1,48 @@
|
|||||||
|
{
|
||||||
|
"ApiSettings": {
|
||||||
|
"BaseUrl": "https://localhost:5000"
|
||||||
|
},
|
||||||
|
"Serilog": {
|
||||||
|
"Using": [
|
||||||
|
"Serilog.Sinks.File",
|
||||||
|
"Serilog.Sinks.Debug",
|
||||||
|
"Serilog.Sinks.Async"
|
||||||
|
],
|
||||||
|
"MinimumLevel": {
|
||||||
|
"Default": "Debug",
|
||||||
|
"Override": {
|
||||||
|
"Microsoft": "Warning",
|
||||||
|
"Microsoft.AspNetCore": "Warning",
|
||||||
|
"System": "Warning"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"WriteTo": [
|
||||||
|
{
|
||||||
|
"Name": "Async",
|
||||||
|
"Args": {
|
||||||
|
"configure": [
|
||||||
|
{
|
||||||
|
"Name": "File",
|
||||||
|
"Args": {
|
||||||
|
"path": "LOG_PATH_PLACEHOLDER",
|
||||||
|
"rollingInterval": "Day",
|
||||||
|
"retainedFileCountLimit": 7,
|
||||||
|
"outputTemplate": "[{Timestamp:yyyy-MM-dd HH:mm:ss.fff zzz}] [{Level:u3}] [ThreadId: {ThreadId}] [{SourceContext}] {Message:lj}{NewLine}{Exception}",
|
||||||
|
"shared": true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "Debug",
|
||||||
|
"Args": {
|
||||||
|
"outputTemplate": "[{Timestamp:yyyy-MM-dd HH:mm:ss.fff zzz}] [{Level:u3}] [ThreadId: {ThreadId}] [{SourceContext}] {Message:lj}{NewLine}{Exception}"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"Enrich": [
|
||||||
|
"FromLogContext"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -26,7 +26,53 @@
|
|||||||
|
|
||||||
<script src="_framework/blazor.webview.js" autostart="false"></script>
|
<script src="_framework/blazor.webview.js" autostart="false"></script>
|
||||||
<script src="_content/NexusReader.UI.Shared/js/d3.v7.min.js"></script>
|
<script src="_content/NexusReader.UI.Shared/js/d3.v7.min.js"></script>
|
||||||
|
<script>
|
||||||
|
window.NexusLogging = {
|
||||||
|
initializeBridge: function (dotNetHelper) {
|
||||||
|
const originalLog = console.log;
|
||||||
|
const originalWarn = console.warn;
|
||||||
|
const originalError = console.error;
|
||||||
|
|
||||||
|
console.log = function (...args) {
|
||||||
|
originalLog.apply(console, args);
|
||||||
|
try {
|
||||||
|
dotNetHelper.invokeMethodAsync('LogJsMessage', 'info', args.map(x => typeof x === 'object' ? JSON.stringify(x) : String(x)).join(' '));
|
||||||
|
} catch (e) {}
|
||||||
|
};
|
||||||
|
|
||||||
|
console.warn = function (...args) {
|
||||||
|
originalWarn.apply(console, args);
|
||||||
|
try {
|
||||||
|
dotNetHelper.invokeMethodAsync('LogJsMessage', 'warn', args.map(x => typeof x === 'object' ? JSON.stringify(x) : String(x)).join(' '));
|
||||||
|
} catch (e) {}
|
||||||
|
};
|
||||||
|
|
||||||
|
console.error = function (...args) {
|
||||||
|
originalError.apply(console, args);
|
||||||
|
try {
|
||||||
|
dotNetHelper.invokeMethodAsync('LogJsMessage', 'error', args.map(x => typeof x === 'object' ? JSON.stringify(x) : String(x)).join(' '));
|
||||||
|
} catch (e) {}
|
||||||
|
};
|
||||||
|
|
||||||
|
window.onerror = function (message, source, lineno, colno, error) {
|
||||||
|
const stack = error ? error.stack : '';
|
||||||
|
try {
|
||||||
|
dotNetHelper.invokeMethodAsync('LogJsMessage', 'error', `${message} at ${source}:${lineno}:${colno}`, stack);
|
||||||
|
} catch (e) {}
|
||||||
|
return false;
|
||||||
|
};
|
||||||
|
|
||||||
|
window.addEventListener('unhandledrejection', function (event) {
|
||||||
|
const reason = event.reason;
|
||||||
|
const message = reason instanceof Error ? reason.message : String(reason);
|
||||||
|
const stack = reason instanceof Error ? reason.stack : '';
|
||||||
|
try {
|
||||||
|
dotNetHelper.invokeMethodAsync('LogJsMessage', 'error', `Unhandled Promise Rejection: ${message}`, stack);
|
||||||
|
} catch (e) {}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
</script>
|
||||||
</body>
|
</body>
|
||||||
|
|
||||||
</html>
|
</html>
|
||||||
|
|||||||
@@ -0,0 +1,76 @@
|
|||||||
|
@using NexusReader.Application.DTOs.AI
|
||||||
|
|
||||||
|
<div class="nexus-citation-container" @onmouseenter="ShowPopup" @onmouseleave="HidePopup">
|
||||||
|
<button type="button" class="nexus-citation-trigger" aria-label="Citation source">
|
||||||
|
<!-- Circular Neon SVG Radar Ping / Stylized Book Icon -->
|
||||||
|
<svg class="neon-radar-svg" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5" stroke-linecap="round" stroke-linejoin="round">
|
||||||
|
<circle cx="12" cy="12" r="10"></circle>
|
||||||
|
<circle cx="12" cy="12" r="6"></circle>
|
||||||
|
<circle cx="12" cy="12" r="2"></circle>
|
||||||
|
</svg>
|
||||||
|
<span class="pulse-ring"></span>
|
||||||
|
</button>
|
||||||
|
|
||||||
|
@if (_isHovered && _citation != null)
|
||||||
|
{
|
||||||
|
<div class="nexus-citation-popup">
|
||||||
|
<div class="popup-header">
|
||||||
|
<span class="book-title"><i class="bi bi-book-half"></i> @_citation.SourceBook</span>
|
||||||
|
@if (!string.IsNullOrEmpty(_citation.Author))
|
||||||
|
{
|
||||||
|
<span class="separator">•</span>
|
||||||
|
<span class="book-author">@_citation.Author</span>
|
||||||
|
}
|
||||||
|
@if (_citation.PageNumber.HasValue)
|
||||||
|
{
|
||||||
|
<span class="separator">•</span>
|
||||||
|
<span class="page-number">Page @_citation.PageNumber.Value</span>
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
<div class="popup-body">
|
||||||
|
<p class="citation-quote">"@_citation.Snippet"</p>
|
||||||
|
</div>
|
||||||
|
<div class="popup-footer">
|
||||||
|
<span class="id-badge">ID: @SourceId.Substring(0, Math.Min(8, SourceId.Length))</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
@code {
|
||||||
|
[Parameter]
|
||||||
|
[EditorRequired]
|
||||||
|
public string SourceId { get; set; } = string.Empty;
|
||||||
|
|
||||||
|
[Parameter]
|
||||||
|
public List<CitationDto>? Citations { get; set; }
|
||||||
|
|
||||||
|
private bool _isHovered;
|
||||||
|
private CitationDto? _citation;
|
||||||
|
|
||||||
|
protected override void OnParametersSet()
|
||||||
|
{
|
||||||
|
_citation = Citations?.FirstOrDefault(c => c.CitationId.Equals(SourceId, System.StringComparison.OrdinalIgnoreCase));
|
||||||
|
|
||||||
|
// If not found in the thread citations, provide a clean fallback so the UI never displays an empty error
|
||||||
|
if (_citation == null)
|
||||||
|
{
|
||||||
|
_citation = new CitationDto
|
||||||
|
{
|
||||||
|
CitationId = SourceId,
|
||||||
|
SourceBook = "Grounded Document Chunk",
|
||||||
|
Snippet = "Context snippet retrieved from vector search node."
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void ShowPopup()
|
||||||
|
{
|
||||||
|
_isHovered = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void HidePopup()
|
||||||
|
{
|
||||||
|
_isHovered = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,148 @@
|
|||||||
|
.nexus-citation-container {
|
||||||
|
position: relative;
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
vertical-align: middle;
|
||||||
|
margin: 0 4px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.nexus-citation-trigger {
|
||||||
|
background: transparent;
|
||||||
|
border: none;
|
||||||
|
padding: 0;
|
||||||
|
margin: 0;
|
||||||
|
cursor: pointer;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
color: #06b6d4; /* Glowing Cyan */
|
||||||
|
width: 20px;
|
||||||
|
height: 20px;
|
||||||
|
position: relative;
|
||||||
|
outline: none;
|
||||||
|
transition: all 0.3s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
.nexus-citation-trigger:hover {
|
||||||
|
color: #00ff99; /* Neon Green on hover */
|
||||||
|
transform: scale(1.2);
|
||||||
|
}
|
||||||
|
|
||||||
|
.neon-radar-svg {
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
filter: drop-shadow(0 0 4px currentColor);
|
||||||
|
animation: radar-spin 8s linear infinite;
|
||||||
|
}
|
||||||
|
|
||||||
|
.pulse-ring {
|
||||||
|
position: absolute;
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
border: 1px solid currentColor;
|
||||||
|
border-radius: 50%;
|
||||||
|
animation: radar-ping 1.5s cubic-bezier(0, 0, 0.2, 1) infinite;
|
||||||
|
opacity: 0;
|
||||||
|
pointer-events: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.nexus-citation-popup {
|
||||||
|
position: absolute;
|
||||||
|
bottom: calc(100% + 10px);
|
||||||
|
left: 50%;
|
||||||
|
transform: translateX(-50%) translateY(5px);
|
||||||
|
width: 320px;
|
||||||
|
padding: 1rem;
|
||||||
|
border-radius: 12px;
|
||||||
|
background: rgba(10, 16, 26, 0.9); /* Premium dark background */
|
||||||
|
border: 1px solid rgba(6, 182, 212, 0.25); /* Cyan border */
|
||||||
|
backdrop-filter: blur(12px);
|
||||||
|
-webkit-backdrop-filter: blur(12px);
|
||||||
|
box-shadow: 0 10px 25px rgba(0, 0, 0, 0.5), 0 0 12px rgba(6, 182, 212, 0.15);
|
||||||
|
z-index: 1000;
|
||||||
|
pointer-events: none;
|
||||||
|
opacity: 0;
|
||||||
|
animation: popup-fade-in 0.2s cubic-bezier(0.16, 1, 0.3, 1) forwards;
|
||||||
|
transform-origin: bottom center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.popup-header {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
gap: 4px;
|
||||||
|
font-size: 0.75rem;
|
||||||
|
font-weight: 700;
|
||||||
|
color: #00ff99; /* Emerald/Neon Green micro-header */
|
||||||
|
text-transform: uppercase;
|
||||||
|
letter-spacing: 0.5px;
|
||||||
|
margin-bottom: 0.5rem;
|
||||||
|
border-bottom: 1px solid rgba(255, 255, 255, 0.05);
|
||||||
|
padding-bottom: 0.35rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.separator {
|
||||||
|
color: rgba(255, 255, 255, 0.3);
|
||||||
|
}
|
||||||
|
|
||||||
|
.book-title {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 4px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.book-author, .page-number {
|
||||||
|
color: rgba(255, 255, 255, 0.6);
|
||||||
|
}
|
||||||
|
|
||||||
|
.popup-body {
|
||||||
|
margin-bottom: 0.5rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.citation-quote {
|
||||||
|
font-size: 0.85rem;
|
||||||
|
line-height: 1.4;
|
||||||
|
color: rgba(255, 255, 255, 0.95);
|
||||||
|
font-style: italic;
|
||||||
|
margin: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.popup-footer {
|
||||||
|
display: flex;
|
||||||
|
justify-content: flex-end;
|
||||||
|
}
|
||||||
|
|
||||||
|
.id-badge {
|
||||||
|
font-size: 0.65rem;
|
||||||
|
color: rgba(255, 255, 255, 0.3);
|
||||||
|
font-family: monospace;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Animations */
|
||||||
|
@keyframes radar-spin {
|
||||||
|
0% { transform: rotate(0deg); }
|
||||||
|
100% { transform: rotate(360deg); }
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes radar-ping {
|
||||||
|
0% {
|
||||||
|
transform: scale(1);
|
||||||
|
opacity: 0.8;
|
||||||
|
}
|
||||||
|
100% {
|
||||||
|
transform: scale(2.2);
|
||||||
|
opacity: 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes popup-fade-in {
|
||||||
|
0% {
|
||||||
|
opacity: 0;
|
||||||
|
transform: translateX(-50%) translateY(8px) scale(0.95);
|
||||||
|
}
|
||||||
|
100% {
|
||||||
|
opacity: 1;
|
||||||
|
transform: translateX(-50%) translateY(0) scale(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,39 +1,347 @@
|
|||||||
@namespace NexusReader.UI.Shared.Components.Atoms
|
@namespace NexusReader.UI.Shared.Components.Atoms
|
||||||
|
@using System.Text.RegularExpressions
|
||||||
|
@using MediatR
|
||||||
|
@using NexusReader.Application.DTOs.AI
|
||||||
|
@using NexusReader.Application.Queries.Library
|
||||||
|
@inject IMediator Mediator
|
||||||
|
@inject IReaderNavigationService NavService
|
||||||
|
@inject IReaderInteractionService InteractionService
|
||||||
|
@inject NavigationManager NavManager
|
||||||
|
@inject AuthenticationStateProvider AuthStateProvider
|
||||||
|
@inject ILogger<NexusSearchBox> Logger
|
||||||
|
@implements IDisposable
|
||||||
|
|
||||||
<div class="nexus-search-container @(IsActive ? "active" : "")">
|
<div class="nexus-search-container @(IsFocused ? "focused" : "") @(HasResults ? "has-results" : "")" @onfocusin="HandleFocusIn" @onfocusout="HandleFocusOut">
|
||||||
<div class="search-wrapper">
|
<div class="search-wrapper">
|
||||||
<i class="nexus-icon @IconClass"></i>
|
<div class="search-icon-container">
|
||||||
<input type="text"
|
@if (_isLoading)
|
||||||
@bind="SearchValue"
|
{
|
||||||
@bind:event="oninput"
|
<div class="neon-spinner"></div>
|
||||||
@onkeypress="HandleKeyPress"
|
}
|
||||||
placeholder="@Placeholder"
|
else
|
||||||
|
{
|
||||||
|
<i class="nexus-icon bi bi-search"></i>
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<input type="text"
|
||||||
|
value="@SearchValue"
|
||||||
|
@oninput="HandleInput"
|
||||||
|
@onkeydown="HandleKeyDown"
|
||||||
|
placeholder="@Placeholder"
|
||||||
class="nexus-search-input" />
|
class="nexus-search-input" />
|
||||||
|
|
||||||
|
<div class="ai-status-indicator" title="Aktywny silnik AI biblioteki">
|
||||||
|
<span class="ai-pulse-dot"></span>
|
||||||
|
</div>
|
||||||
|
|
||||||
@if (!string.IsNullOrEmpty(SearchValue))
|
@if (!string.IsNullOrEmpty(SearchValue))
|
||||||
{
|
{
|
||||||
<button class="clear-btn" @onclick="ClearSearch">×</button>
|
<button type="button" class="clear-btn" @onclick="ClearSearch" aria-label="Wyczyść wyszukiwanie">×</button>
|
||||||
}
|
}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
@if (_isDropdownOpen && (!string.IsNullOrEmpty(SearchValue) || _isLoading || _results.Any() || _searchError != null))
|
||||||
|
{
|
||||||
|
<div class="search-dropdown glass-panel">
|
||||||
|
@if (_isLoading)
|
||||||
|
{
|
||||||
|
<div class="dropdown-state-container">
|
||||||
|
<div class="neon-spinner-large"></div>
|
||||||
|
<span class="state-text">Analizowanie biblioteki semantycznej...</span>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
else if (_searchError != null)
|
||||||
|
{
|
||||||
|
<div class="dropdown-state-container error">
|
||||||
|
<i class="bi bi-exclamation-triangle-fill error-icon"></i>
|
||||||
|
<span class="state-text">@_searchError</span>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
else if (_results.Any())
|
||||||
|
{
|
||||||
|
<div class="dropdown-results-list">
|
||||||
|
@foreach (var result in _results)
|
||||||
|
{
|
||||||
|
<div class="result-card" @onclick="() => HandleResultClick(result)">
|
||||||
|
<div class="result-header">
|
||||||
|
<span class="relevance-badge">@(Math.Round(result.RelevanceScore * 100))% Trafności</span>
|
||||||
|
@if (!string.IsNullOrEmpty(result.SourceBookTitle))
|
||||||
|
{
|
||||||
|
<span class="source-title" title="@result.SourceBookTitle">w <strong>@result.SourceBookTitle</strong></span>
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
<div class="result-snippet">
|
||||||
|
@((MarkupString)HighlightQueryWords(result.Snippet, SearchValue))
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
else if (!string.IsNullOrEmpty(SearchValue))
|
||||||
|
{
|
||||||
|
<div class="dropdown-state-container empty">
|
||||||
|
<i class="bi bi-search empty-icon"></i>
|
||||||
|
<span class="state-text">Brak wyników dla zapytania.</span>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@code {
|
@code {
|
||||||
[Parameter] public string Placeholder { get; set; } = "Search your library...";
|
[Parameter] public string Placeholder { get; set; } = "Zapytaj swoją bibliotekę AI...";
|
||||||
[Parameter] public string IconClass { get; set; } = "bi bi-search";
|
|
||||||
[Parameter] public EventCallback<string> OnSearch { get; set; }
|
[Parameter] public EventCallback<string> OnSearch { get; set; }
|
||||||
|
[Parameter] public int Limit { get; set; } = 5;
|
||||||
private string SearchValue { get; set; } = string.Empty;
|
|
||||||
private bool IsActive => !string.IsNullOrEmpty(SearchValue);
|
|
||||||
|
|
||||||
private async Task HandleKeyPress(KeyboardEventArgs e)
|
private string SearchValue { get; set; } = string.Empty;
|
||||||
|
private bool IsFocused { get; set; }
|
||||||
|
private bool HasResults => _results.Any() && _isDropdownOpen;
|
||||||
|
|
||||||
|
private List<SemanticSearchResultDto> _results = new();
|
||||||
|
private bool _isLoading;
|
||||||
|
private string? _searchError;
|
||||||
|
private bool _isDropdownOpen;
|
||||||
|
private bool _disposed;
|
||||||
|
|
||||||
|
private CancellationTokenSource? _searchCts;
|
||||||
|
|
||||||
|
private async Task HandleInput(ChangeEventArgs e)
|
||||||
{
|
{
|
||||||
if (e.Key == "Enter")
|
SearchValue = e.Value?.ToString() ?? string.Empty;
|
||||||
|
_searchError = null;
|
||||||
|
|
||||||
|
if (string.IsNullOrWhiteSpace(SearchValue))
|
||||||
{
|
{
|
||||||
await OnSearch.InvokeAsync(SearchValue);
|
_results.Clear();
|
||||||
|
_isDropdownOpen = false;
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
_isDropdownOpen = true;
|
||||||
|
|
||||||
|
// Cancel previous search in-flight
|
||||||
|
_searchCts?.Cancel();
|
||||||
|
_searchCts?.Dispose();
|
||||||
|
_searchCts = new CancellationTokenSource();
|
||||||
|
|
||||||
|
var token = _searchCts.Token;
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
// Debounce for 300ms
|
||||||
|
await Task.Delay(300, token);
|
||||||
|
await PerformSearchAsync(token);
|
||||||
|
}
|
||||||
|
catch (TaskCanceledException)
|
||||||
|
{
|
||||||
|
// Typing continued, search cancelled
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task PerformSearchAsync(CancellationToken token)
|
||||||
|
{
|
||||||
|
_isLoading = true;
|
||||||
|
_searchError = null;
|
||||||
|
if (!_disposed)
|
||||||
|
{
|
||||||
|
await InvokeAsync(StateHasChanged);
|
||||||
|
}
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var authState = await AuthStateProvider.GetAuthenticationStateAsync();
|
||||||
|
var tenantId = authState.User.FindFirst("TenantId")?.Value ?? "global";
|
||||||
|
|
||||||
|
var result = await Mediator.Send(new SearchLibrarySemanticallyQuery(SearchValue, tenantId, Limit), token);
|
||||||
|
if (token.IsCancellationRequested || _disposed) return;
|
||||||
|
|
||||||
|
if (result.IsSuccess)
|
||||||
|
{
|
||||||
|
_results = result.Value ?? new List<SemanticSearchResultDto>();
|
||||||
|
_searchError = null;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
_results.Clear();
|
||||||
|
_searchError = result.Errors.FirstOrDefault()?.Message ?? "Nie udało się wykonać wyszukiwania.";
|
||||||
|
Logger.LogWarning("Semantic search returned errors: {Errors}", string.Join(", ", result.Errors.Select(e => e.Message)));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
if (!token.IsCancellationRequested && !_disposed)
|
||||||
|
{
|
||||||
|
_results.Clear();
|
||||||
|
_searchError = "Wystąpił nieoczekiwany błąd podczas wyszukiwania.";
|
||||||
|
Logger.LogError(ex, "Unexpected error during semantic search for query: {Query}", SearchValue);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
finally
|
||||||
|
{
|
||||||
|
if (!token.IsCancellationRequested && !_disposed)
|
||||||
|
{
|
||||||
|
_isLoading = false;
|
||||||
|
await InvokeAsync(StateHasChanged);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task HandleResultClick(SemanticSearchResultDto result)
|
||||||
|
{
|
||||||
|
_isDropdownOpen = false;
|
||||||
|
|
||||||
|
// 1. Resolve Ebook ID
|
||||||
|
Guid? ebookId = null;
|
||||||
|
if (result.Metadata != null)
|
||||||
|
{
|
||||||
|
foreach (var key in new[] { "ebookId", "ebook_id", "EbookId", "Ebook_Id" })
|
||||||
|
{
|
||||||
|
if (result.Metadata.TryGetValue(key, out var val) && val != null)
|
||||||
|
{
|
||||||
|
if (Guid.TryParse(val.ToString(), out var g))
|
||||||
|
{
|
||||||
|
ebookId = g;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (ebookId == null || ebookId == Guid.Empty)
|
||||||
|
{
|
||||||
|
ebookId = NavService.CurrentEbookId;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (ebookId == null || ebookId == Guid.Empty)
|
||||||
|
{
|
||||||
|
Logger.LogWarning("Could not resolve ebook ID from search result metadata.");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 2. Resolve Chapter Index
|
||||||
|
int chapterIndex = 0;
|
||||||
|
if (result.Metadata != null)
|
||||||
|
{
|
||||||
|
foreach (var key in new[] { "chapterIndex", "chapter_index", "ChapterIndex", "chapter" })
|
||||||
|
{
|
||||||
|
if (result.Metadata.TryGetValue(key, out var val) && val != null)
|
||||||
|
{
|
||||||
|
if (int.TryParse(val.ToString(), out var parsedInt))
|
||||||
|
{
|
||||||
|
chapterIndex = parsedInt;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 3. Resolve Block ID
|
||||||
|
string? blockId = null;
|
||||||
|
if (result.Metadata != null)
|
||||||
|
{
|
||||||
|
foreach (var key in new[] { "blockId", "block_id", "BlockId", "nodeId", "node_id", "NodeId", "id" })
|
||||||
|
{
|
||||||
|
if (result.Metadata.TryGetValue(key, out var val) && val != null)
|
||||||
|
{
|
||||||
|
blockId = val.ToString();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (string.IsNullOrEmpty(blockId))
|
||||||
|
{
|
||||||
|
blockId = result.ContentHash;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 4. Set pending scroll and navigate
|
||||||
|
NavService.PendingScrollBlockId = blockId;
|
||||||
|
|
||||||
|
if (NavService.CurrentEbookId == ebookId.Value && NavService.CurrentChapterIndex == chapterIndex)
|
||||||
|
{
|
||||||
|
// Same chapter - scroll and highlight immediately
|
||||||
|
if (!string.IsNullOrEmpty(blockId))
|
||||||
|
{
|
||||||
|
await InteractionService.RequestScrollToBlock(blockId);
|
||||||
|
await InteractionService.RequestHighlightBlock(blockId);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// Different chapter or book - perform routing
|
||||||
|
NavService.SetBook(ebookId.Value, chapterIndex);
|
||||||
|
NavManager.NavigateTo($"/reader/{ebookId.Value}?chapter={chapterIndex}");
|
||||||
|
}
|
||||||
|
|
||||||
|
// Invoke the optional callback for parent components
|
||||||
|
await OnSearch.InvokeAsync(SearchValue);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void HandleKeyDown(KeyboardEventArgs e)
|
||||||
|
{
|
||||||
|
if (e.Key == "Escape")
|
||||||
|
{
|
||||||
|
_isDropdownOpen = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void HandleFocusIn()
|
||||||
|
{
|
||||||
|
IsFocused = true;
|
||||||
|
_isDropdownOpen = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task HandleFocusOut()
|
||||||
|
{
|
||||||
|
IsFocused = false;
|
||||||
|
// Delay slightly to allow click handlers on result cards to execute
|
||||||
|
await Task.Delay(200);
|
||||||
|
if (_disposed) return;
|
||||||
|
_isDropdownOpen = false;
|
||||||
|
StateHasChanged();
|
||||||
}
|
}
|
||||||
|
|
||||||
private void ClearSearch()
|
private void ClearSearch()
|
||||||
{
|
{
|
||||||
SearchValue = string.Empty;
|
SearchValue = string.Empty;
|
||||||
|
_results.Clear();
|
||||||
|
_searchError = null;
|
||||||
|
_isDropdownOpen = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
private string HighlightQueryWords(string text, string query)
|
||||||
|
{
|
||||||
|
if (string.IsNullOrWhiteSpace(text))
|
||||||
|
return string.Empty;
|
||||||
|
|
||||||
|
var escapedText = System.Net.WebUtility.HtmlEncode(text);
|
||||||
|
|
||||||
|
if (string.IsNullOrWhiteSpace(query))
|
||||||
|
return escapedText;
|
||||||
|
|
||||||
|
var words = query.Split(new[] { ' ' }, StringSplitOptions.RemoveEmptyEntries)
|
||||||
|
.Where(w => w.Length > 2)
|
||||||
|
.Select(Regex.Escape);
|
||||||
|
|
||||||
|
if (!words.Any())
|
||||||
|
return escapedText;
|
||||||
|
|
||||||
|
var pattern = "(" + string.Join("|", words) + ")";
|
||||||
|
try
|
||||||
|
{
|
||||||
|
return Regex.Replace(escapedText, pattern, "<mark class=\"search-highlight\">$1</mark>", RegexOptions.IgnoreCase);
|
||||||
|
}
|
||||||
|
catch
|
||||||
|
{
|
||||||
|
return escapedText;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Dispose()
|
||||||
|
{
|
||||||
|
_disposed = true;
|
||||||
|
_searchCts?.Cancel();
|
||||||
|
_searchCts?.Dispose();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,57 +1,309 @@
|
|||||||
.nexus-search-container {
|
.nexus-search-container {
|
||||||
|
position: relative;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
max-width: 500px;
|
max-width: 600px;
|
||||||
margin: 1rem auto;
|
margin: 1.5rem auto;
|
||||||
transition: all 0.3s ease;
|
font-family: var(--nexus-font-sans), 'Inter', sans-serif;
|
||||||
|
z-index: 1000;
|
||||||
}
|
}
|
||||||
|
|
||||||
.search-wrapper {
|
.search-wrapper {
|
||||||
position: relative;
|
position: relative;
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
background: var(--nexus-card, #141414);
|
background: rgba(255, 255, 255, 0.03);
|
||||||
border: 1px solid rgba(255, 255, 255, 0.1);
|
backdrop-filter: blur(10px);
|
||||||
border-radius: 12px;
|
-webkit-backdrop-filter: blur(10px);
|
||||||
padding: 0.5rem 1rem;
|
border: 1px solid rgba(255, 255, 255, 0.08);
|
||||||
transition: border-color 0.3s ease, box-shadow 0.3s ease;
|
border-radius: 14px;
|
||||||
|
padding: 0.65rem 1.1rem;
|
||||||
|
transition: all 0.3s cubic-bezier(0.25, 0.8, 0.25, 1);
|
||||||
|
box-shadow: 0 4px 12px rgba(0, 0, 0, 0.3);
|
||||||
}
|
}
|
||||||
|
|
||||||
.nexus-search-container.active .search-wrapper,
|
/* Focused state: glowing neon border matching other dashboard components */
|
||||||
.search-wrapper:focus-within {
|
.nexus-search-container.focused .search-wrapper {
|
||||||
border-color: var(--nexus-neon, #00ff99);
|
background: rgba(255, 255, 255, 0.05);
|
||||||
box-shadow: 0 0 15px rgba(0, 255, 153, 0.2);
|
border-color: var(--nexus-neon);
|
||||||
|
box-shadow: 0 8px 24px rgba(0, 0, 0, 0.5), 0 0 15px rgba(0, 255, 153, 0.25);
|
||||||
|
}
|
||||||
|
|
||||||
|
.search-icon-container {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
width: 20px;
|
||||||
|
height: 20px;
|
||||||
|
margin-right: 0.85rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
.nexus-icon {
|
.nexus-icon {
|
||||||
color: rgba(255, 255, 255, 0.5);
|
color: rgba(255, 255, 255, 0.45);
|
||||||
margin-right: 0.75rem;
|
font-size: 1.25rem;
|
||||||
font-size: 1.1rem;
|
transition: color 0.3s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
.nexus-search-container.focused .nexus-icon {
|
||||||
|
color: var(--nexus-neon);
|
||||||
}
|
}
|
||||||
|
|
||||||
.nexus-search-input {
|
.nexus-search-input {
|
||||||
flex: 1;
|
flex: 1;
|
||||||
background: transparent;
|
background: transparent;
|
||||||
border: none;
|
border: none;
|
||||||
color: white;
|
color: #ffffff;
|
||||||
font-family: 'Inter', sans-serif;
|
font-size: 1rem;
|
||||||
font-size: 0.95rem;
|
font-weight: 400;
|
||||||
outline: none;
|
outline: none;
|
||||||
|
padding: 0;
|
||||||
|
width: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
.nexus-search-input::placeholder {
|
.nexus-search-input::placeholder {
|
||||||
color: rgba(255, 255, 255, 0.3);
|
color: rgba(255, 255, 255, 0.35);
|
||||||
|
font-style: italic;
|
||||||
|
transition: color 0.3s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
.nexus-search-container.focused .nexus-search-input::placeholder {
|
||||||
|
color: rgba(255, 255, 255, 0.5);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Pulsing neon-green AI status indicator */
|
||||||
|
.ai-status-indicator {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
margin: 0 0.75rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.ai-pulse-dot {
|
||||||
|
width: 8px;
|
||||||
|
height: 8px;
|
||||||
|
background-color: var(--nexus-neon);
|
||||||
|
border-radius: 50%;
|
||||||
|
display: inline-block;
|
||||||
|
position: relative;
|
||||||
|
box-shadow: 0 0 8px var(--nexus-neon);
|
||||||
|
}
|
||||||
|
|
||||||
|
.ai-pulse-dot::after {
|
||||||
|
content: '';
|
||||||
|
position: absolute;
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
background-color: var(--nexus-neon);
|
||||||
|
border-radius: 50%;
|
||||||
|
z-index: -1;
|
||||||
|
animation: pulse 2s infinite ease-in-out;
|
||||||
}
|
}
|
||||||
|
|
||||||
.clear-btn {
|
.clear-btn {
|
||||||
background: transparent;
|
background: transparent;
|
||||||
border: none;
|
border: none;
|
||||||
color: rgba(255, 255, 255, 0.4);
|
color: rgba(255, 255, 255, 0.4);
|
||||||
font-size: 1.2rem;
|
font-size: 1.5rem;
|
||||||
|
line-height: 1;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
padding: 0 0.5rem;
|
padding: 0 0.25rem;
|
||||||
transition: color 0.2s ease;
|
margin-left: 0.5rem;
|
||||||
|
transition: color 0.2s ease, transform 0.2s ease;
|
||||||
}
|
}
|
||||||
|
|
||||||
.clear-btn:hover {
|
.clear-btn:hover {
|
||||||
color: var(--nexus-neon, #00ff99);
|
color: #ff3b30;
|
||||||
|
transform: scale(1.1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Frosted glass results container */
|
||||||
|
.search-dropdown {
|
||||||
|
position: absolute;
|
||||||
|
top: calc(100% + 8px);
|
||||||
|
left: 0;
|
||||||
|
right: 0;
|
||||||
|
background: rgba(18, 18, 18, 0.9);
|
||||||
|
backdrop-filter: blur(20px) saturate(160%);
|
||||||
|
-webkit-backdrop-filter: blur(20px) saturate(160%);
|
||||||
|
border: 1px solid rgba(255, 255, 255, 0.08);
|
||||||
|
border-radius: 14px;
|
||||||
|
box-shadow: 0 12px 36px rgba(0, 0, 0, 0.6), 0 0 20px rgba(0, 255, 153, 0.05);
|
||||||
|
max-height: 420px;
|
||||||
|
overflow-y: auto;
|
||||||
|
overflow-x: hidden;
|
||||||
|
scrollbar-width: thin;
|
||||||
|
scrollbar-color: rgba(255, 255, 255, 0.15) transparent;
|
||||||
|
transition: all 0.3s cubic-bezier(0.25, 0.8, 0.25, 1);
|
||||||
|
animation: slideDown 0.25s cubic-bezier(0.16, 1, 0.3, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
.search-dropdown::-webkit-scrollbar {
|
||||||
|
width: 6px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.search-dropdown::-webkit-scrollbar-thumb {
|
||||||
|
background: rgba(255, 255, 255, 0.15);
|
||||||
|
border-radius: 3px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.search-dropdown::-webkit-scrollbar-thumb:hover {
|
||||||
|
background: var(--nexus-neon);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* In-flight spinners */
|
||||||
|
.neon-spinner {
|
||||||
|
width: 16px;
|
||||||
|
height: 16px;
|
||||||
|
border: 2px solid rgba(0, 255, 153, 0.15);
|
||||||
|
border-top: 2px solid var(--nexus-neon);
|
||||||
|
border-radius: 50%;
|
||||||
|
animation: spin 0.75s linear infinite;
|
||||||
|
}
|
||||||
|
|
||||||
|
.neon-spinner-large {
|
||||||
|
width: 32px;
|
||||||
|
height: 32px;
|
||||||
|
border: 3px solid rgba(255, 255, 255, 0.05);
|
||||||
|
border-top: 3px solid var(--nexus-neon);
|
||||||
|
border-radius: 50%;
|
||||||
|
animation: spin 0.8s cubic-bezier(0.5, 0.1, 0.5, 0.9) infinite;
|
||||||
|
margin-bottom: 1rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.dropdown-state-container {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
padding: 2.5rem 1.5rem;
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.state-text {
|
||||||
|
font-size: 0.95rem;
|
||||||
|
color: rgba(255, 255, 255, 0.65);
|
||||||
|
font-weight: 300;
|
||||||
|
}
|
||||||
|
|
||||||
|
.error-icon, .empty-icon {
|
||||||
|
font-size: 2rem;
|
||||||
|
margin-bottom: 0.75rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.error-icon {
|
||||||
|
color: #ff3b30;
|
||||||
|
text-shadow: 0 0 10px rgba(255, 59, 48, 0.4);
|
||||||
|
}
|
||||||
|
|
||||||
|
.empty-icon {
|
||||||
|
color: rgba(255, 255, 255, 0.2);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Results Cards list */
|
||||||
|
.dropdown-results-list {
|
||||||
|
padding: 0.5rem;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 0.4rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.result-card {
|
||||||
|
padding: 0.95rem 1.1rem;
|
||||||
|
background: rgba(255, 255, 255, 0.02);
|
||||||
|
border: 1px solid rgba(255, 255, 255, 0.04);
|
||||||
|
border-radius: 10px;
|
||||||
|
cursor: pointer;
|
||||||
|
transition: all 0.2s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
.result-card:hover {
|
||||||
|
background: rgba(0, 255, 153, 0.05);
|
||||||
|
border-color: rgba(0, 255, 153, 0.2);
|
||||||
|
transform: translateY(-1px);
|
||||||
|
box-shadow: 0 4px 12px rgba(0, 0, 0, 0.3);
|
||||||
|
}
|
||||||
|
|
||||||
|
.result-header {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: space-between;
|
||||||
|
margin-bottom: 0.5rem;
|
||||||
|
font-size: 0.8rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.relevance-badge {
|
||||||
|
background: rgba(0, 255, 153, 0.1);
|
||||||
|
color: var(--nexus-neon);
|
||||||
|
border: 1px solid rgba(0, 255, 153, 0.25);
|
||||||
|
border-radius: 6px;
|
||||||
|
padding: 0.15rem 0.45rem;
|
||||||
|
font-weight: 600;
|
||||||
|
letter-spacing: 0.02em;
|
||||||
|
text-shadow: 0 0 4px rgba(0, 255, 153, 0.25);
|
||||||
|
}
|
||||||
|
|
||||||
|
.source-title {
|
||||||
|
color: rgba(255, 255, 255, 0.5);
|
||||||
|
max-width: 60%;
|
||||||
|
overflow: hidden;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
white-space: nowrap;
|
||||||
|
}
|
||||||
|
|
||||||
|
.source-title strong {
|
||||||
|
color: rgba(255, 255, 255, 0.85);
|
||||||
|
}
|
||||||
|
|
||||||
|
.result-snippet {
|
||||||
|
font-size: 0.88rem;
|
||||||
|
line-height: 1.45;
|
||||||
|
color: rgba(255, 255, 255, 0.78);
|
||||||
|
font-weight: 300;
|
||||||
|
display: -webkit-box;
|
||||||
|
-webkit-line-clamp: 3;
|
||||||
|
-webkit-box-orient: vertical;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Markup highlights */
|
||||||
|
::deep mark.search-highlight {
|
||||||
|
background: rgba(0, 255, 153, 0.2);
|
||||||
|
color: var(--nexus-neon);
|
||||||
|
border-bottom: 1px solid var(--nexus-neon);
|
||||||
|
padding: 0.05rem 0.15rem;
|
||||||
|
border-radius: 3px;
|
||||||
|
font-weight: 500;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Animations */
|
||||||
|
@keyframes pulse {
|
||||||
|
0% {
|
||||||
|
transform: scale(1);
|
||||||
|
box-shadow: 0 0 0 0 rgba(0, 255, 153, 0.7);
|
||||||
|
}
|
||||||
|
70% {
|
||||||
|
transform: scale(1.6);
|
||||||
|
box-shadow: 0 0 0 6px rgba(0, 255, 153, 0);
|
||||||
|
}
|
||||||
|
100% {
|
||||||
|
transform: scale(1);
|
||||||
|
box-shadow: 0 0 0 0 rgba(0, 255, 153, 0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes spin {
|
||||||
|
to {
|
||||||
|
transform: rotate(360deg);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes slideDown {
|
||||||
|
from {
|
||||||
|
opacity: 0;
|
||||||
|
transform: translateY(-8px);
|
||||||
|
}
|
||||||
|
to {
|
||||||
|
opacity: 1;
|
||||||
|
transform: translateY(0);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,9 +2,14 @@
|
|||||||
@using NexusReader.Application.Queries.Quiz
|
@using NexusReader.Application.Queries.Quiz
|
||||||
@using NexusReader.Application.Commands.Quiz
|
@using NexusReader.Application.Commands.Quiz
|
||||||
@using NexusReader.Application.Abstractions.Services
|
@using NexusReader.Application.Abstractions.Services
|
||||||
|
@using NexusReader.UI.Shared.Components.Atoms
|
||||||
|
@using NexusReader.UI.Shared.Services
|
||||||
@inject IMediator Mediator
|
@inject IMediator Mediator
|
||||||
@inject IPlatformService PlatformService
|
@inject IPlatformService PlatformService
|
||||||
@inject IQuizStateService QuizService
|
@inject IQuizStateService QuizService
|
||||||
|
@inject IIdentityService IdentityService
|
||||||
|
@inject IKnowledgeGraphService GraphService
|
||||||
|
@inject KnowledgeCoordinator Coordinator
|
||||||
|
|
||||||
<div class="knowledge-check">
|
<div class="knowledge-check">
|
||||||
<div class="quiz-header">
|
<div class="quiz-header">
|
||||||
@@ -12,10 +17,33 @@
|
|||||||
<button class="expand-btn">⌵</button>
|
<button class="expand-btn">⌵</button>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@if (QuizService.IsHydrating)
|
@if (QuizService.IsHydrating || _isGenerating)
|
||||||
{
|
{
|
||||||
<div class="loading-state shimmer">Skanowanie wiedzy przez AI...</div>
|
<div class="loading-state shimmer">Skanowanie wiedzy przez AI...</div>
|
||||||
}
|
}
|
||||||
|
else if (_isSubmitted)
|
||||||
|
{
|
||||||
|
<div class="submitted-container">
|
||||||
|
<div class="success-icon-wrapper">
|
||||||
|
<NexusIcon Name="check" Size="48" Class="success-glow" />
|
||||||
|
</div>
|
||||||
|
<h3 class="submitted-title">Gratulacje!</h3>
|
||||||
|
<p class="submitted-text">Sprawdzian zakończony pomyślnie. Twój wynik został zapisany w bazie danych.</p>
|
||||||
|
|
||||||
|
<div class="score-card">
|
||||||
|
<div class="score-main">
|
||||||
|
<span class="score-num">@_score</span>
|
||||||
|
<span class="score-divider">/</span>
|
||||||
|
<span class="score-total">@_totalQuestions</span>
|
||||||
|
</div>
|
||||||
|
<div class="score-percent">@((int)_percentage)% poprawnych odpowiedzi</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<button class="reset-quiz-btn" @onclick="CloseQuiz">
|
||||||
|
<span>ZAKOŃCZ</span>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
else if (QuizService.CurrentQuiz != null)
|
else if (QuizService.CurrentQuiz != null)
|
||||||
{
|
{
|
||||||
<div class="quiz-body">
|
<div class="quiz-body">
|
||||||
@@ -41,17 +69,45 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
<div class="quiz-footer">
|
<div class="quiz-footer">
|
||||||
<button class="submit-btn" disabled="@(!AllQuestionsAnswered())">Wyślij</button>
|
<button class="submit-btn" disabled="@(!AllQuestionsAnswered() || _isSubmitting)" @onclick="SubmitQuizAsync">
|
||||||
|
@if (_isSubmitting)
|
||||||
|
{
|
||||||
|
<span>Zapisywanie...</span>
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
<span>Wyślij</span>
|
||||||
|
}
|
||||||
|
</button>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
}
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
<div class="empty-quiz-state">
|
||||||
|
<div class="empty-icon-wrapper">
|
||||||
|
<NexusIcon Name="robot" Size="48" Class="neon-glow" />
|
||||||
|
</div>
|
||||||
|
<h3 class="empty-title">Brak Aktywnego Quizu</h3>
|
||||||
|
<p class="empty-text">Generuj spersonalizowany sprawdzian wiedzy na podstawie bieżącego rozdziału książki.</p>
|
||||||
|
|
||||||
|
<button class="generate-quiz-btn" @onclick="GenerateChapterQuizAsync" disabled="@(string.IsNullOrWhiteSpace(Coordinator.CurrentFullPageContent))">
|
||||||
|
<span>GENERUJ QUIZ DLA ROZDZIAŁU</span>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
|
||||||
@code {
|
@code {
|
||||||
[Parameter] public string ContextBlockId { get; set; } = string.Empty;
|
[Parameter] public string ContextBlockId { get; set; } = string.Empty;
|
||||||
|
|
||||||
private Dictionary<QuizQuestionDto, (int SelectedIndex, bool IsCorrect)> _states = new();
|
private Dictionary<QuizQuestionDto, (int SelectedIndex, bool IsCorrect)> _states = new();
|
||||||
|
private bool _isSubmitting = false;
|
||||||
|
private bool _isSubmitted = false;
|
||||||
|
private bool _isGenerating = false;
|
||||||
|
private int _score = 0;
|
||||||
|
private int _totalQuestions = 0;
|
||||||
|
private double _percentage = 0.0;
|
||||||
|
|
||||||
protected override void OnInitialized()
|
protected override void OnInitialized()
|
||||||
{
|
{
|
||||||
@@ -65,6 +121,24 @@
|
|||||||
QuizService.OnQuizUpdated -= HandleUpdate;
|
QuizService.OnQuizUpdated -= HandleUpdate;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private async Task GenerateChapterQuizAsync()
|
||||||
|
{
|
||||||
|
if (_isGenerating || string.IsNullOrWhiteSpace(Coordinator.CurrentFullPageContent)) return;
|
||||||
|
|
||||||
|
_isGenerating = true;
|
||||||
|
StateHasChanged();
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
await Coordinator.RequestSummaryAndQuizAsync(Coordinator.CurrentFullPageContent);
|
||||||
|
}
|
||||||
|
finally
|
||||||
|
{
|
||||||
|
_isGenerating = false;
|
||||||
|
StateHasChanged();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private async Task SelectOptionAsync(QuizQuestionDto question, int index)
|
private async Task SelectOptionAsync(QuizQuestionDto question, int index)
|
||||||
{
|
{
|
||||||
if (_states.ContainsKey(question)) return;
|
if (_states.ContainsKey(question)) return;
|
||||||
@@ -90,6 +164,67 @@
|
|||||||
return QuizService.CurrentQuiz != null && _states.Count == QuizService.CurrentQuiz.Questions.Count;
|
return QuizService.CurrentQuiz != null && _states.Count == QuizService.CurrentQuiz.Questions.Count;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private async Task SubmitQuizAsync()
|
||||||
|
{
|
||||||
|
if (QuizService.CurrentQuiz == null || !AllQuestionsAnswered() || _isSubmitting) return;
|
||||||
|
|
||||||
|
_isSubmitting = true;
|
||||||
|
StateHasChanged();
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
_score = _states.Values.Count(s => s.IsCorrect);
|
||||||
|
_totalQuestions = QuizService.CurrentQuiz.Questions.Count;
|
||||||
|
_percentage = _totalQuestions > 0 ? ((double)_score / _totalQuestions) * 100 : 0.0;
|
||||||
|
|
||||||
|
string topic = "Quiz wiedzy";
|
||||||
|
var graph = GraphService.CurrentGraphData;
|
||||||
|
if (graph != null && !string.IsNullOrEmpty(QuizService.CurrentQuizBlockId))
|
||||||
|
{
|
||||||
|
var node = graph.Nodes.FirstOrDefault(n => n.Id == QuizService.CurrentQuizBlockId);
|
||||||
|
if (node != null && !string.IsNullOrEmpty(node.Label))
|
||||||
|
{
|
||||||
|
topic = $"Test: {node.Label}";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var profileResult = await IdentityService.GetProfileAsync();
|
||||||
|
if (profileResult.IsSuccess && profileResult.Value != null)
|
||||||
|
{
|
||||||
|
var userId = profileResult.Value.UserId;
|
||||||
|
|
||||||
|
var cmd = new SubmitQuizResultCommand(userId, topic, _score, _totalQuestions);
|
||||||
|
var result = await Mediator.Send(cmd);
|
||||||
|
|
||||||
|
if (result.IsSuccess)
|
||||||
|
{
|
||||||
|
IdentityService.ClearCache();
|
||||||
|
_isSubmitted = true;
|
||||||
|
await PlatformService.VibrateSuccessAsync();
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
await PlatformService.VibrateErrorAsync();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch
|
||||||
|
{
|
||||||
|
await PlatformService.VibrateErrorAsync();
|
||||||
|
}
|
||||||
|
finally
|
||||||
|
{
|
||||||
|
_isSubmitting = false;
|
||||||
|
StateHasChanged();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void CloseQuiz()
|
||||||
|
{
|
||||||
|
_isSubmitted = false;
|
||||||
|
_states.Clear();
|
||||||
|
QuizService.SetQuiz(null, null);
|
||||||
|
}
|
||||||
|
|
||||||
private string GetBlockClass(QuizQuestionDto question)
|
private string GetBlockClass(QuizQuestionDto question)
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -121,3 +121,217 @@
|
|||||||
0% { background-position: -200% 0; }
|
0% { background-position: -200% 0; }
|
||||||
100% { background-position: 200% 0; }
|
100% { background-position: 200% 0; }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.option-revealed-correct {
|
||||||
|
border-color: #00ff99 !important;
|
||||||
|
background: rgba(0, 255, 153, 0.08) !important;
|
||||||
|
box-shadow: 0 0 8px rgba(0, 255, 153, 0.15);
|
||||||
|
}
|
||||||
|
|
||||||
|
.option-faded {
|
||||||
|
opacity: 0.45;
|
||||||
|
}
|
||||||
|
|
||||||
|
.submitted-container {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
text-align: center;
|
||||||
|
padding: 2rem 1rem;
|
||||||
|
animation: fadeIn 0.4s ease-out;
|
||||||
|
}
|
||||||
|
|
||||||
|
.success-icon-wrapper {
|
||||||
|
background: rgba(0, 255, 153, 0.1);
|
||||||
|
border: 1px solid rgba(0, 255, 153, 0.3);
|
||||||
|
border-radius: 50%;
|
||||||
|
width: 80px;
|
||||||
|
height: 80px;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
margin-bottom: 1.5rem;
|
||||||
|
box-shadow: 0 0 20px rgba(0, 255, 153, 0.15);
|
||||||
|
}
|
||||||
|
|
||||||
|
.success-glow {
|
||||||
|
color: var(--nexus-neon, #00ff99);
|
||||||
|
filter: drop-shadow(0 0 8px var(--nexus-neon, #00ff99));
|
||||||
|
}
|
||||||
|
|
||||||
|
.submitted-title {
|
||||||
|
font-size: 1.5rem;
|
||||||
|
font-weight: 700;
|
||||||
|
color: #fff;
|
||||||
|
margin-bottom: 0.5rem;
|
||||||
|
letter-spacing: -0.5px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.submitted-text {
|
||||||
|
font-size: 0.9rem;
|
||||||
|
color: #888;
|
||||||
|
margin-bottom: 2rem;
|
||||||
|
line-height: 1.5;
|
||||||
|
}
|
||||||
|
|
||||||
|
.score-card {
|
||||||
|
background: rgba(255, 255, 255, 0.02);
|
||||||
|
border: 1px solid rgba(255, 255, 255, 0.05);
|
||||||
|
border-radius: 16px;
|
||||||
|
padding: 1.5rem 2.5rem;
|
||||||
|
margin-bottom: 2rem;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: center;
|
||||||
|
backdrop-filter: blur(10px);
|
||||||
|
}
|
||||||
|
|
||||||
|
.score-main {
|
||||||
|
display: flex;
|
||||||
|
align-items: baseline;
|
||||||
|
gap: 0.2rem;
|
||||||
|
margin-bottom: 0.5rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.score-num {
|
||||||
|
font-size: 3rem;
|
||||||
|
font-weight: 800;
|
||||||
|
color: var(--nexus-neon, #00ff99);
|
||||||
|
line-height: 1;
|
||||||
|
text-shadow: 0 0 15px rgba(0, 255, 153, 0.3);
|
||||||
|
}
|
||||||
|
|
||||||
|
.score-divider {
|
||||||
|
font-size: 1.8rem;
|
||||||
|
color: #444;
|
||||||
|
}
|
||||||
|
|
||||||
|
.score-total {
|
||||||
|
font-size: 1.8rem;
|
||||||
|
font-weight: 600;
|
||||||
|
color: #fff;
|
||||||
|
}
|
||||||
|
|
||||||
|
.score-percent {
|
||||||
|
font-size: 0.85rem;
|
||||||
|
color: #666;
|
||||||
|
text-transform: uppercase;
|
||||||
|
letter-spacing: 1px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.reset-quiz-btn {
|
||||||
|
padding: 0.8rem 3rem;
|
||||||
|
background: transparent;
|
||||||
|
border: 1px solid rgba(255, 255, 255, 0.15);
|
||||||
|
border-radius: 30px;
|
||||||
|
color: #fff;
|
||||||
|
font-size: 0.9rem;
|
||||||
|
font-weight: 600;
|
||||||
|
cursor: pointer;
|
||||||
|
transition: all 0.3s cubic-bezier(0.4, 0, 0.2, 1);
|
||||||
|
letter-spacing: 0.5px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.reset-quiz-btn:hover {
|
||||||
|
background: rgba(255, 255, 255, 0.05);
|
||||||
|
border-color: #fff;
|
||||||
|
box-shadow: 0 0 15px rgba(255, 255, 255, 0.1);
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes fadeIn {
|
||||||
|
from {
|
||||||
|
opacity: 0;
|
||||||
|
transform: translateY(10px);
|
||||||
|
}
|
||||||
|
to {
|
||||||
|
opacity: 1;
|
||||||
|
transform: translateY(0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.empty-quiz-state {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
text-align: center;
|
||||||
|
padding: 2.5rem 1rem;
|
||||||
|
animation: fadeIn 0.4s ease-out;
|
||||||
|
}
|
||||||
|
|
||||||
|
.empty-icon-wrapper {
|
||||||
|
background: rgba(0, 255, 153, 0.03);
|
||||||
|
border: 1px solid rgba(0, 255, 153, 0.15);
|
||||||
|
border-radius: 50%;
|
||||||
|
width: 80px;
|
||||||
|
height: 80px;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
margin-bottom: 1.5rem;
|
||||||
|
box-shadow: 0 0 30px rgba(0, 255, 153, 0.05);
|
||||||
|
transition: all 0.3s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
.empty-quiz-state:hover .empty-icon-wrapper {
|
||||||
|
background: rgba(0, 255, 153, 0.08);
|
||||||
|
border-color: rgba(0, 255, 153, 0.4);
|
||||||
|
box-shadow: 0 0 35px rgba(0, 255, 153, 0.15);
|
||||||
|
transform: scale(1.05);
|
||||||
|
}
|
||||||
|
|
||||||
|
.neon-glow {
|
||||||
|
color: var(--nexus-neon, #00ff99);
|
||||||
|
filter: drop-shadow(0 0 6px var(--nexus-neon, #00ff99));
|
||||||
|
}
|
||||||
|
|
||||||
|
.empty-title {
|
||||||
|
font-size: 1.3rem;
|
||||||
|
font-weight: 700;
|
||||||
|
color: #fff;
|
||||||
|
margin-bottom: 0.5rem;
|
||||||
|
letter-spacing: -0.3px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.empty-text {
|
||||||
|
font-size: 0.9rem;
|
||||||
|
color: #888;
|
||||||
|
margin-bottom: 2rem;
|
||||||
|
line-height: 1.5;
|
||||||
|
max-width: 280px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.generate-quiz-btn {
|
||||||
|
padding: 0.85rem 2rem;
|
||||||
|
background: rgba(0, 255, 153, 0.08);
|
||||||
|
border: 1px solid var(--nexus-neon, #00ff99);
|
||||||
|
border-radius: 30px;
|
||||||
|
color: var(--nexus-neon, #00ff99);
|
||||||
|
font-size: 0.85rem;
|
||||||
|
font-weight: 700;
|
||||||
|
cursor: pointer;
|
||||||
|
transition: all 0.3s cubic-bezier(0.4, 0, 0.2, 1);
|
||||||
|
letter-spacing: 0.8px;
|
||||||
|
text-shadow: 0 0 10px rgba(0, 255, 153, 0.3);
|
||||||
|
box-shadow: 0 0 15px rgba(0, 255, 153, 0.1);
|
||||||
|
}
|
||||||
|
|
||||||
|
.generate-quiz-btn:not(:disabled):hover {
|
||||||
|
background: var(--nexus-neon, #00ff99);
|
||||||
|
color: #000;
|
||||||
|
box-shadow: 0 0 25px rgba(0, 255, 153, 0.4);
|
||||||
|
transform: translateY(-2px);
|
||||||
|
text-shadow: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.generate-quiz-btn:disabled {
|
||||||
|
opacity: 0.4;
|
||||||
|
cursor: not-allowed;
|
||||||
|
border-color: rgba(255, 255, 255, 0.15);
|
||||||
|
background: rgba(255, 255, 255, 0.02);
|
||||||
|
color: #666;
|
||||||
|
text-shadow: none;
|
||||||
|
box-shadow: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -2,12 +2,14 @@
|
|||||||
@using NexusReader.Application.Abstractions.Services
|
@using NexusReader.Application.Abstractions.Services
|
||||||
@using NexusReader.Application.Queries.Reader
|
@using NexusReader.Application.Queries.Reader
|
||||||
@using NexusReader.Application.Commands.Library
|
@using NexusReader.Application.Commands.Library
|
||||||
|
@using NexusReader.UI.Shared.Services
|
||||||
@using System.Net.Http.Json
|
@using System.Net.Http.Json
|
||||||
@inject IEpubMetadataExtractor MetadataExtractor
|
@inject IEpubMetadataExtractor MetadataExtractor
|
||||||
@inject ILogger<BookIngestionModal> Logger
|
@inject ILogger<BookIngestionModal> Logger
|
||||||
@inject HttpClient Http
|
@inject HttpClient Http
|
||||||
@inject IReaderNavigationService ReaderNavigation
|
@inject IReaderNavigationService ReaderNavigation
|
||||||
@inject IJSRuntime JSRuntime
|
@inject IJSRuntime JSRuntime
|
||||||
|
@inject ISyncService SyncService
|
||||||
@implements IAsyncDisposable
|
@implements IAsyncDisposable
|
||||||
|
|
||||||
@if (IsOpen)
|
@if (IsOpen)
|
||||||
@@ -16,20 +18,23 @@
|
|||||||
<div class="modal-content glass-panel" @onclick:stopPropagation>
|
<div class="modal-content glass-panel" @onclick:stopPropagation>
|
||||||
<div class="modal-header">
|
<div class="modal-header">
|
||||||
<h2>Add New Book</h2>
|
<h2>Add New Book</h2>
|
||||||
<button class="close-btn" @onclick="CloseModal">
|
@if (!IsIngesting && !IsIndexing)
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><line x1="18" y1="6" x2="6" y2="18"></line><line x1="6" y1="6" x2="18" y2="18"></line></svg>
|
{
|
||||||
</button>
|
<button class="close-btn" @onclick="CloseModal">
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><line x1="18" y1="6" x2="6" y2="18"></line><line x1="6" y1="6" x2="18" y2="18"></line></svg>
|
||||||
|
</button>
|
||||||
|
}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="modal-body">
|
<div class="modal-body">
|
||||||
<div class="parsing-state shimmer" style="@(IsParsing ? "display:flex;" : "display:none;")">
|
<div class="parsing-state shimmer" style="@(IsParsing && !IsIndexing ? "display:flex;" : "display:none;")">
|
||||||
<div class="shimmer-content">
|
<div class="shimmer-content">
|
||||||
<div class="spinner"></div>
|
<div class="spinner"></div>
|
||||||
<p>Scanning metadata...</p>
|
<p>Scanning metadata...</p>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="verification-state" style="@(IsVerifying && !IsParsing ? "display:flex;" : "display:none;")">
|
<div class="verification-state" style="@(IsVerifying && !IsParsing && !IsIndexing ? "display:flex;" : "display:none;")">
|
||||||
@if (Metadata != null)
|
@if (Metadata != null)
|
||||||
{
|
{
|
||||||
<div class="verification-layout">
|
<div class="verification-layout">
|
||||||
@@ -74,7 +79,7 @@
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="upload-state @(_isDragging ? "drag-over" : "")"
|
<div class="upload-state @(_isDragging ? "drag-over" : "")"
|
||||||
style="@(!IsParsing && !IsVerifying ? "display:flex;" : "display:none;")"
|
style="@(!IsParsing && !IsVerifying && !IsIndexing ? "display:flex;" : "display:none;")"
|
||||||
@ondragenter="OnDragEnter"
|
@ondragenter="OnDragEnter"
|
||||||
@ondragleave="OnDragLeave">
|
@ondragleave="OnDragLeave">
|
||||||
<div class="drop-zone">
|
<div class="drop-zone">
|
||||||
@@ -87,6 +92,18 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<div class="indexing-state" style="@(IsIndexing ? "display:flex;" : "display:none;")">
|
||||||
|
<div class="indexing-content">
|
||||||
|
<div class="spinner"></div>
|
||||||
|
<h3>Nexus AI Indexing</h3>
|
||||||
|
<p class="status-msg">@IngestionStatusMessage</p>
|
||||||
|
<div class="progress-bar-container">
|
||||||
|
<div class="progress-bar-fill" style="width: @((IngestionProgressPercent * 100).ToString("F0"))%"></div>
|
||||||
|
</div>
|
||||||
|
<span class="percent">@((IngestionProgressPercent * 100).ToString("F0"))%</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
|
||||||
@if (!string.IsNullOrEmpty(ErrorMessage))
|
@if (!string.IsNullOrEmpty(ErrorMessage))
|
||||||
{
|
{
|
||||||
@@ -118,15 +135,61 @@
|
|||||||
private bool IsParsing { get; set; }
|
private bool IsParsing { get; set; }
|
||||||
private bool IsVerifying { get; set; }
|
private bool IsVerifying { get; set; }
|
||||||
private bool IsIngesting { get; set; }
|
private bool IsIngesting { get; set; }
|
||||||
|
private bool IsIndexing { get; set; }
|
||||||
|
private string IngestionStatusMessage { get; set; } = "Initializing...";
|
||||||
|
private double IngestionProgressPercent { get; set; }
|
||||||
|
private Guid IngestedBookId { get; set; } = Guid.Empty;
|
||||||
private LocalEpubMetadata? Metadata { get; set; }
|
private LocalEpubMetadata? Metadata { get; set; }
|
||||||
private string? ErrorMessage { get; set; }
|
private string? ErrorMessage { get; set; }
|
||||||
private byte[]? _epubBytes;
|
private byte[]? _epubBytes;
|
||||||
|
private bool _disposed;
|
||||||
|
|
||||||
// Allow up to 50 MB
|
// Allow up to 50 MB
|
||||||
private const long MaxFileSize = 50 * 1024 * 1024;
|
private const long MaxFileSize = 50 * 1024 * 1024;
|
||||||
|
|
||||||
|
protected override async Task OnInitializedAsync()
|
||||||
|
{
|
||||||
|
await SyncService.InitializeAsync();
|
||||||
|
SyncService.OnIngestionProgressReceived += HandleIngestionProgress;
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task HandleIngestionProgress(string message, double progress)
|
||||||
|
{
|
||||||
|
if (_disposed) return;
|
||||||
|
if (!IsIndexing) return;
|
||||||
|
|
||||||
|
IngestionStatusMessage = message;
|
||||||
|
IngestionProgressPercent = progress;
|
||||||
|
|
||||||
|
if (!_disposed)
|
||||||
|
{
|
||||||
|
await InvokeAsync(StateHasChanged);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (progress >= 1.0)
|
||||||
|
{
|
||||||
|
// Give the user a moment to see the completion message
|
||||||
|
await Task.Delay(2500);
|
||||||
|
|
||||||
|
if (_disposed) return;
|
||||||
|
|
||||||
|
// Now close the modal and navigate to the book
|
||||||
|
if (IngestedBookId != Guid.Empty)
|
||||||
|
{
|
||||||
|
var bookId = IngestedBookId;
|
||||||
|
await InvokeAsync(async () => {
|
||||||
|
if (_disposed) return;
|
||||||
|
await CloseModal();
|
||||||
|
ReaderNavigation.NavigateToBook(bookId);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private async Task CloseModal()
|
private async Task CloseModal()
|
||||||
{
|
{
|
||||||
|
if (IsIngesting || IsIndexing) return;
|
||||||
|
|
||||||
IsOpen = false;
|
IsOpen = false;
|
||||||
Reset();
|
Reset();
|
||||||
await IsOpenChanged.InvokeAsync(false);
|
await IsOpenChanged.InvokeAsync(false);
|
||||||
@@ -137,6 +200,10 @@
|
|||||||
IsParsing = false;
|
IsParsing = false;
|
||||||
IsVerifying = false;
|
IsVerifying = false;
|
||||||
IsIngesting = false;
|
IsIngesting = false;
|
||||||
|
IsIndexing = false;
|
||||||
|
IngestionStatusMessage = "Initializing...";
|
||||||
|
IngestionProgressPercent = 0.0;
|
||||||
|
IngestedBookId = Guid.Empty;
|
||||||
Metadata = null;
|
Metadata = null;
|
||||||
ErrorMessage = null;
|
ErrorMessage = null;
|
||||||
_isDragging = false;
|
_isDragging = false;
|
||||||
@@ -168,10 +235,12 @@
|
|||||||
using var stream = file.OpenReadStream(MaxFileSize);
|
using var stream = file.OpenReadStream(MaxFileSize);
|
||||||
using var memoryStream = new MemoryStream();
|
using var memoryStream = new MemoryStream();
|
||||||
await stream.CopyToAsync(memoryStream);
|
await stream.CopyToAsync(memoryStream);
|
||||||
|
if (_disposed) return;
|
||||||
_epubBytes = memoryStream.ToArray();
|
_epubBytes = memoryStream.ToArray();
|
||||||
|
|
||||||
memoryStream.Position = 0;
|
memoryStream.Position = 0;
|
||||||
var result = await MetadataExtractor.ExtractMetadataAsync(memoryStream);
|
var result = await MetadataExtractor.ExtractMetadataAsync(memoryStream);
|
||||||
|
if (_disposed) return;
|
||||||
|
|
||||||
if (result.IsSuccess)
|
if (result.IsSuccess)
|
||||||
{
|
{
|
||||||
@@ -186,12 +255,18 @@
|
|||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
Logger.LogError(ex, "Error uploading EPUB");
|
Logger.LogError(ex, "Error uploading EPUB");
|
||||||
ErrorMessage = $"An unexpected error occurred: {ex.Message}";
|
if (!_disposed)
|
||||||
|
{
|
||||||
|
ErrorMessage = $"An unexpected error occurred: {ex.Message}";
|
||||||
|
}
|
||||||
}
|
}
|
||||||
finally
|
finally
|
||||||
{
|
{
|
||||||
IsParsing = false;
|
if (!_disposed)
|
||||||
StateHasChanged();
|
{
|
||||||
|
IsParsing = false;
|
||||||
|
StateHasChanged();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -214,39 +289,55 @@
|
|||||||
);
|
);
|
||||||
|
|
||||||
var response = await Http.PostAsJsonAsync("api/library/ingest", request);
|
var response = await Http.PostAsJsonAsync("api/library/ingest", request);
|
||||||
|
if (_disposed) return;
|
||||||
|
|
||||||
if (response.IsSuccessStatusCode)
|
if (response.IsSuccessStatusCode)
|
||||||
{
|
{
|
||||||
var result = await response.Content.ReadFromJsonAsync<IngestResult>();
|
var result = await response.Content.ReadFromJsonAsync<IngestResult>();
|
||||||
|
if (_disposed) return;
|
||||||
if (result != null)
|
if (result != null)
|
||||||
{
|
{
|
||||||
await CloseModal();
|
IngestedBookId = result.Id;
|
||||||
ReaderNavigation.NavigateToBook(result.Id);
|
IsVerifying = false;
|
||||||
|
IsIngesting = false;
|
||||||
|
IsIndexing = true;
|
||||||
|
IngestionStatusMessage = "Book saved! Starting background indexing...";
|
||||||
|
IngestionProgressPercent = 0.0;
|
||||||
|
StateHasChanged();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
ErrorMessage = await response.Content.ReadAsStringAsync();
|
ErrorMessage = await response.Content.ReadAsStringAsync();
|
||||||
|
IsIngesting = false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
Logger.LogError(ex, "Error during ingestion");
|
Logger.LogError(ex, "Error during ingestion");
|
||||||
ErrorMessage = "Failed to save book to library. Please try again.";
|
if (!_disposed)
|
||||||
|
{
|
||||||
|
ErrorMessage = "Failed to save book to library. Please try again.";
|
||||||
|
IsIngesting = false;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
finally
|
finally
|
||||||
{
|
{
|
||||||
IsIngesting = false;
|
if (!_disposed)
|
||||||
StateHasChanged();
|
{
|
||||||
|
StateHasChanged();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private record IngestResult(Guid Id);
|
private record IngestResult(Guid Id);
|
||||||
|
|
||||||
public ValueTask DisposeAsync()
|
public async ValueTask DisposeAsync()
|
||||||
{
|
{
|
||||||
|
_disposed = true;
|
||||||
|
SyncService.OnIngestionProgressReceived -= HandleIngestionProgress;
|
||||||
// Clear the large byte array so it is eligible for GC even if the component is cached.
|
// Clear the large byte array so it is eligible for GC even if the component is cached.
|
||||||
_epubBytes = null;
|
_epubBytes = null;
|
||||||
return ValueTask.CompletedTask;
|
await ValueTask.CompletedTask;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -377,6 +377,72 @@
|
|||||||
animation: spin 0.8s linear infinite;
|
animation: spin 0.8s linear infinite;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Indexing State */
|
||||||
|
.indexing-state {
|
||||||
|
flex: 1;
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
|
border-radius: 12px;
|
||||||
|
background: rgba(255, 255, 255, 0.02);
|
||||||
|
border: 1px solid rgba(255, 255, 255, 0.05);
|
||||||
|
box-shadow: inset 0 0 12px rgba(255, 255, 255, 0.02);
|
||||||
|
position: relative;
|
||||||
|
overflow: hidden;
|
||||||
|
padding: 2rem;
|
||||||
|
animation: fadeIn 0.4s ease-out;
|
||||||
|
}
|
||||||
|
|
||||||
|
.indexing-content {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: center;
|
||||||
|
text-align: center;
|
||||||
|
width: 100%;
|
||||||
|
gap: 1.25rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.indexing-content h3 {
|
||||||
|
margin: 0;
|
||||||
|
font-size: 1.25rem;
|
||||||
|
color: var(--nexus-neon, #00ffaa);
|
||||||
|
text-shadow: 0 0 10px rgba(0, 255, 153, 0.2);
|
||||||
|
letter-spacing: 0.5px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.status-msg {
|
||||||
|
margin: 0;
|
||||||
|
font-size: 0.9rem;
|
||||||
|
color: var(--nexus-text-muted, #888);
|
||||||
|
min-height: 2.5rem;
|
||||||
|
line-height: 1.4;
|
||||||
|
}
|
||||||
|
|
||||||
|
.progress-bar-container {
|
||||||
|
width: 100%;
|
||||||
|
height: 8px;
|
||||||
|
background: rgba(255, 255, 255, 0.05);
|
||||||
|
border-radius: 4px;
|
||||||
|
overflow: hidden;
|
||||||
|
position: relative;
|
||||||
|
border: 1px solid rgba(255, 255, 255, 0.05);
|
||||||
|
}
|
||||||
|
|
||||||
|
.progress-bar-fill {
|
||||||
|
height: 100%;
|
||||||
|
background: linear-gradient(90deg, var(--nexus-neon, #00ffaa) 0%, #00b3ff 100%);
|
||||||
|
box-shadow: 0 0 10px rgba(0, 255, 153, 0.4);
|
||||||
|
border-radius: 4px;
|
||||||
|
transition: width 0.4s cubic-bezier(0.4, 0, 0.2, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
.percent {
|
||||||
|
font-family: var(--nexus-font-mono, monospace);
|
||||||
|
font-size: 1.1rem;
|
||||||
|
font-weight: 700;
|
||||||
|
color: var(--nexus-text);
|
||||||
|
}
|
||||||
|
|
||||||
@keyframes fadeIn {
|
@keyframes fadeIn {
|
||||||
from { opacity: 0; }
|
from { opacity: 0; }
|
||||||
to { opacity: 1; }
|
to { opacity: 1; }
|
||||||
|
|||||||
@@ -67,6 +67,7 @@
|
|||||||
private bool _isJsInitialized;
|
private bool _isJsInitialized;
|
||||||
private ElementReference _containerRef;
|
private ElementReference _containerRef;
|
||||||
private bool _isInteractive;
|
private bool _isInteractive;
|
||||||
|
private string? _currentActiveBlockId;
|
||||||
|
|
||||||
protected override async Task OnInitializedAsync()
|
protected override async Task OnInitializedAsync()
|
||||||
{
|
{
|
||||||
@@ -143,6 +144,7 @@
|
|||||||
[JSInvokable]
|
[JSInvokable]
|
||||||
public async Task HandleBlockReached(string blockId, string content)
|
public async Task HandleBlockReached(string blockId, string content)
|
||||||
{
|
{
|
||||||
|
_currentActiveBlockId = blockId;
|
||||||
await Coordinator.OnBlockReachedAsync(blockId, content);
|
await Coordinator.OnBlockReachedAsync(blockId, content);
|
||||||
|
|
||||||
if (ViewModel != null)
|
if (ViewModel != null)
|
||||||
@@ -160,8 +162,15 @@
|
|||||||
|
|
||||||
private async Task HandleSyncProgressReceived(string blockId, DateTime timestamp)
|
private async Task HandleSyncProgressReceived(string blockId, DateTime timestamp)
|
||||||
{
|
{
|
||||||
|
if (string.IsNullOrEmpty(blockId) || blockId == _currentActiveBlockId)
|
||||||
|
{
|
||||||
|
Logger.LogDebug("[Sync] Received progress {BlockId} is empty or matches active block. Ignoring scroll.", blockId);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
Logger.LogInformation("[Sync] Received progress from another device: block {BlockId} at {Timestamp}", blockId, timestamp);
|
Logger.LogInformation("[Sync] Received progress from another device: block {BlockId} at {Timestamp}", blockId, timestamp);
|
||||||
|
|
||||||
|
_currentActiveBlockId = blockId;
|
||||||
await ScrollToNodeAsync(blockId);
|
await ScrollToNodeAsync(blockId);
|
||||||
await InvokeAsync(StateHasChanged);
|
await InvokeAsync(StateHasChanged);
|
||||||
}
|
}
|
||||||
@@ -211,6 +220,8 @@
|
|||||||
|
|
||||||
private async Task LoadChapterAsync(int index)
|
private async Task LoadChapterAsync(int index)
|
||||||
{
|
{
|
||||||
|
await Coordinator.ClearAsync();
|
||||||
|
_isJsInitialized = false; // Reset JS initialization to re-bind the scroll observer to new DOM elements!
|
||||||
_isLoadingChapter = true;
|
_isLoadingChapter = true;
|
||||||
StatusMessage = "Wczytywanie treści...";
|
StatusMessage = "Wczytywanie treści...";
|
||||||
StateHasChanged();
|
StateHasChanged();
|
||||||
@@ -247,6 +258,18 @@
|
|||||||
|
|
||||||
_isLoadingChapter = false;
|
_isLoadingChapter = false;
|
||||||
StateHasChanged();
|
StateHasChanged();
|
||||||
|
|
||||||
|
if (result.IsSuccess && !string.IsNullOrEmpty(NavigationService.PendingScrollBlockId))
|
||||||
|
{
|
||||||
|
var targetBlockId = NavigationService.PendingScrollBlockId;
|
||||||
|
NavigationService.PendingScrollBlockId = null; // Clear it to prevent multiple scrolls
|
||||||
|
_currentActiveBlockId = targetBlockId;
|
||||||
|
|
||||||
|
// Give the browser slightly more than one frame to render the loaded blocks
|
||||||
|
await Task.Delay(150);
|
||||||
|
await ScrollToNodeAsync(targetBlockId);
|
||||||
|
await InteractionService.RequestHighlightBlock(targetBlockId);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task ScrollToNodeAsync(string id)
|
public async Task ScrollToNodeAsync(string id)
|
||||||
|
|||||||
@@ -3,10 +3,13 @@
|
|||||||
@using NexusReader.UI.Shared.Services
|
@using NexusReader.UI.Shared.Services
|
||||||
@using NexusReader.UI.Shared.Components.Molecules
|
@using NexusReader.UI.Shared.Components.Molecules
|
||||||
@using NexusReader.UI.Shared.Components.Organisms
|
@using NexusReader.UI.Shared.Components.Organisms
|
||||||
|
@using NexusReader.Application.Queries.Graph
|
||||||
@using Microsoft.Extensions.Logging
|
@using Microsoft.Extensions.Logging
|
||||||
@inject IPlatformService PlatformService
|
@inject IPlatformService PlatformService
|
||||||
@inject IFocusModeService FocusMode
|
@inject IFocusModeService FocusMode
|
||||||
@inject IQuizStateService QuizService
|
@inject IQuizStateService QuizService
|
||||||
|
@inject IReaderInteractionService InteractionService
|
||||||
|
@inject IKnowledgeGraphService GraphService
|
||||||
@inject IJSRuntime JS
|
@inject IJSRuntime JS
|
||||||
@inject IIdentityService IdentityService
|
@inject IIdentityService IdentityService
|
||||||
@inject NavigationManager NavigationManager
|
@inject NavigationManager NavigationManager
|
||||||
@@ -41,13 +44,92 @@
|
|||||||
<button class="close-btn">×</button>
|
<button class="close-btn">×</button>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="intelligence-scroll-area">
|
@if (_activeTab == SidebarTab.Knowledge)
|
||||||
@if (!_isMobile)
|
{
|
||||||
{
|
<div class="intelligence-scroll-area stacked-layout">
|
||||||
<KnowledgeGraph />
|
@if (!_isMobile)
|
||||||
}
|
{
|
||||||
<KnowledgeCheck />
|
<div class="visual-workspace">
|
||||||
</div>
|
<KnowledgeGraph />
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
|
||||||
|
<div class="contextual-intelligence-panel">
|
||||||
|
<div class="panel-header">
|
||||||
|
<NexusIcon Name="brain" Size="18" Class="neon-accent-icon" />
|
||||||
|
<span class="panel-title">Contextual Intelligence Panel</span>
|
||||||
|
</div>
|
||||||
|
<div class="panel-body">
|
||||||
|
@if (_selectedNode != null)
|
||||||
|
{
|
||||||
|
<div class="node-details">
|
||||||
|
<div class="node-header-section">
|
||||||
|
<span class="node-group-badge @(_selectedNode.Group.ToLower())">@(_selectedNode.Group.ToUpper())</span>
|
||||||
|
<h3 class="node-label">@_selectedNode.Label</h3>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
@if (!string.IsNullOrEmpty(_selectedNode.Description))
|
||||||
|
{
|
||||||
|
<div class="detail-section">
|
||||||
|
<p class="node-description">@_selectedNode.Description</p>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
|
||||||
|
@if (!string.IsNullOrEmpty(_selectedNode.Summary))
|
||||||
|
{
|
||||||
|
<div class="detail-section summary-section">
|
||||||
|
<h4 class="section-title neon-sub-header">Podsumowanie</h4>
|
||||||
|
<p class="node-summary">@_selectedNode.Summary</p>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
|
||||||
|
@if (_selectedNode.KeyTerms != null && _selectedNode.KeyTerms.Any())
|
||||||
|
{
|
||||||
|
<div class="detail-section key-terms-section">
|
||||||
|
<h4 class="section-title neon-sub-header">Kluczowe Pojęcia</h4>
|
||||||
|
<ul class="key-terms-list">
|
||||||
|
@foreach (var term in _selectedNode.KeyTerms)
|
||||||
|
{
|
||||||
|
<li class="key-term-item">
|
||||||
|
<span class="term-bullet">•</span>
|
||||||
|
<span class="term-text">@term</span>
|
||||||
|
</li>
|
||||||
|
}
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
<div class="no-node-selected">
|
||||||
|
<div class="placeholder-glow"></div>
|
||||||
|
<p class="placeholder-text">Wybierz węzeł na wykresie, aby wyświetlić szczegóły architektoniczne.</p>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="sidebar-footer">
|
||||||
|
<button class="open-quiz-btn neon-glow-btn @(QuizService.HasNewQuiz ? "quiz-pulse-btn" : "")" @onclick="() => SetActiveTab(SidebarTab.Quiz)">
|
||||||
|
<NexusIcon Name="quiz" Size="18" />
|
||||||
|
<span>OPEN KNOWLEDGE QUIZ</span>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
<div class="intelligence-scroll-area quiz-layout">
|
||||||
|
<div class="quiz-nav">
|
||||||
|
<button class="back-to-graph-btn" @onclick="() => SetActiveTab(SidebarTab.Knowledge)">
|
||||||
|
<NexusIcon Name="arrow-left" Size="16" />
|
||||||
|
<span>← Powrót do wykresu</span>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
<KnowledgeCheck />
|
||||||
|
</div>
|
||||||
|
}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</Authorized>
|
</Authorized>
|
||||||
@@ -67,6 +149,16 @@
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
@code {
|
@code {
|
||||||
|
private enum SidebarTab
|
||||||
|
{
|
||||||
|
Knowledge,
|
||||||
|
Quiz
|
||||||
|
}
|
||||||
|
|
||||||
|
private SidebarTab _activeTab = SidebarTab.Knowledge;
|
||||||
|
private string? _selectedNodeId;
|
||||||
|
private GraphNodeDto? _selectedNode;
|
||||||
|
|
||||||
private string _platformClass = "platform-desktop";
|
private string _platformClass = "platform-desktop";
|
||||||
private bool _isMobile = false;
|
private bool _isMobile = false;
|
||||||
|
|
||||||
@@ -74,6 +166,10 @@
|
|||||||
{
|
{
|
||||||
FocusMode.OnFocusModeChanged += HandleUpdate;
|
FocusMode.OnFocusModeChanged += HandleUpdate;
|
||||||
QuizService.OnQuizUpdated += HandleUpdate;
|
QuizService.OnQuizUpdated += HandleUpdate;
|
||||||
|
QuizService.OnQuizRequested += HandleQuizRequestedAsync;
|
||||||
|
|
||||||
|
InteractionService.OnNodeSelected += HandleNodeSelectedAsync;
|
||||||
|
GraphService.OnGraphUpdated += HandleGraphUpdatedAsync;
|
||||||
|
|
||||||
var context = PlatformService.GetDeviceContext();
|
var context = PlatformService.GetDeviceContext();
|
||||||
if (context.IsSuccess)
|
if (context.IsSuccess)
|
||||||
@@ -88,7 +184,34 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void SetActiveTab(SidebarTab tab)
|
||||||
|
{
|
||||||
|
_activeTab = tab;
|
||||||
|
StateHasChanged();
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task HandleQuizRequestedAsync(string blockId)
|
||||||
|
{
|
||||||
|
_activeTab = SidebarTab.Quiz;
|
||||||
|
await InvokeAsync(StateHasChanged);
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task HandleNodeSelectedAsync(string nodeId)
|
||||||
|
{
|
||||||
|
_selectedNodeId = nodeId;
|
||||||
|
if (GraphService.CurrentGraphData != null)
|
||||||
|
{
|
||||||
|
_selectedNode = GraphService.CurrentGraphData.Nodes.FirstOrDefault(n => n.Id == nodeId);
|
||||||
|
}
|
||||||
|
await InvokeAsync(StateHasChanged);
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task HandleGraphUpdatedAsync()
|
||||||
|
{
|
||||||
|
_selectedNodeId = null;
|
||||||
|
_selectedNode = null;
|
||||||
|
await InvokeAsync(StateHasChanged);
|
||||||
|
}
|
||||||
|
|
||||||
protected override async Task OnAfterRenderAsync(bool firstRender)
|
protected override async Task OnAfterRenderAsync(bool firstRender)
|
||||||
{
|
{
|
||||||
@@ -112,5 +235,8 @@
|
|||||||
{
|
{
|
||||||
FocusMode.OnFocusModeChanged -= HandleUpdate;
|
FocusMode.OnFocusModeChanged -= HandleUpdate;
|
||||||
QuizService.OnQuizUpdated -= HandleUpdate;
|
QuizService.OnQuizUpdated -= HandleUpdate;
|
||||||
|
QuizService.OnQuizRequested -= HandleQuizRequestedAsync;
|
||||||
|
InteractionService.OnNodeSelected -= HandleNodeSelectedAsync;
|
||||||
|
GraphService.OnGraphUpdated -= HandleGraphUpdatedAsync;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -152,4 +152,319 @@ main {
|
|||||||
0% { filter: drop-shadow(0 0 2px var(--nexus-neon)); transform: scale(1); }
|
0% { filter: drop-shadow(0 0 2px var(--nexus-neon)); transform: scale(1); }
|
||||||
50% { filter: drop-shadow(0 0 10px var(--nexus-neon)); transform: scale(1.1); }
|
50% { filter: drop-shadow(0 0 10px var(--nexus-neon)); transform: scale(1.1); }
|
||||||
100% { filter: drop-shadow(0 0 2px var(--nexus-neon)); transform: scale(1); }
|
100% { filter: drop-shadow(0 0 2px var(--nexus-neon)); transform: scale(1); }
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Contextual Intelligence Panel Layout */
|
||||||
|
.stacked-layout {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
height: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.visual-workspace {
|
||||||
|
flex-shrink: 0;
|
||||||
|
border-bottom: 1px solid rgba(255, 255, 255, 0.05);
|
||||||
|
}
|
||||||
|
|
||||||
|
.contextual-intelligence-panel {
|
||||||
|
flex: 1;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
background: rgba(13, 13, 13, 0.6);
|
||||||
|
border-top: 1px solid rgba(255, 255, 255, 0.03);
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
.panel-header {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 0.5rem;
|
||||||
|
padding: 0.75rem 1.25rem;
|
||||||
|
background: rgba(255, 255, 255, 0.01);
|
||||||
|
border-bottom: 1px solid rgba(255, 255, 255, 0.03);
|
||||||
|
}
|
||||||
|
|
||||||
|
.neon-accent-icon {
|
||||||
|
color: var(--nexus-neon, #00f0ff);
|
||||||
|
filter: drop-shadow(0 0 5px var(--nexus-neon, #00f0ff));
|
||||||
|
}
|
||||||
|
|
||||||
|
.panel-title {
|
||||||
|
font-family: var(--nexus-font-sans, "Outfit", sans-serif);
|
||||||
|
font-size: 0.75rem;
|
||||||
|
text-transform: uppercase;
|
||||||
|
letter-spacing: 0.1em;
|
||||||
|
color: rgba(255, 255, 255, 0.5);
|
||||||
|
font-weight: 600;
|
||||||
|
}
|
||||||
|
|
||||||
|
.panel-body {
|
||||||
|
flex: 1;
|
||||||
|
overflow-y: auto;
|
||||||
|
padding: 1.25rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.no-node-selected {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
height: 100%;
|
||||||
|
min-height: 150px;
|
||||||
|
text-align: center;
|
||||||
|
color: rgba(255, 255, 255, 0.35);
|
||||||
|
padding: 2rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.placeholder-glow {
|
||||||
|
width: 48px;
|
||||||
|
height: 48px;
|
||||||
|
border-radius: 50%;
|
||||||
|
background: radial-gradient(circle, rgba(0, 240, 255, 0.15) 0%, transparent 70%);
|
||||||
|
animation: glow-pulse 2s infinite ease-in-out;
|
||||||
|
margin-bottom: 1rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes glow-pulse {
|
||||||
|
0% { transform: scale(0.9); opacity: 0.5; }
|
||||||
|
50% { transform: scale(1.1); opacity: 1; }
|
||||||
|
100% { transform: scale(0.9); opacity: 0.5; }
|
||||||
|
}
|
||||||
|
|
||||||
|
.placeholder-text {
|
||||||
|
font-size: 0.85rem;
|
||||||
|
line-height: 1.4;
|
||||||
|
font-weight: 300;
|
||||||
|
}
|
||||||
|
|
||||||
|
.node-details {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 1.25rem;
|
||||||
|
animation: fade-in 0.3s ease-out;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes fade-in {
|
||||||
|
from { opacity: 0; transform: translateY(5px); }
|
||||||
|
to { opacity: 1; transform: translateY(0); }
|
||||||
|
}
|
||||||
|
|
||||||
|
.node-header-section {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 0.5rem;
|
||||||
|
border-bottom: 1px solid rgba(255, 255, 255, 0.05);
|
||||||
|
padding-bottom: 0.75rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.node-group-badge {
|
||||||
|
align-self: flex-start;
|
||||||
|
font-size: 0.65rem;
|
||||||
|
font-weight: 700;
|
||||||
|
padding: 0.15rem 0.5rem;
|
||||||
|
border-radius: 4px;
|
||||||
|
letter-spacing: 0.08em;
|
||||||
|
border: 1px solid transparent;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Badge specific styling matching category theme colors */
|
||||||
|
.node-group-badge.rule {
|
||||||
|
background: rgba(244, 63, 94, 0.1);
|
||||||
|
color: #f43f5e;
|
||||||
|
border-color: rgba(244, 63, 94, 0.3);
|
||||||
|
}
|
||||||
|
|
||||||
|
.node-group-badge.definition {
|
||||||
|
background: rgba(234, 179, 8, 0.1);
|
||||||
|
color: #eab308;
|
||||||
|
border-color: rgba(234, 179, 8, 0.3);
|
||||||
|
}
|
||||||
|
|
||||||
|
.node-group-badge.table {
|
||||||
|
background: rgba(168, 85, 247, 0.1);
|
||||||
|
color: #a855f7;
|
||||||
|
border-color: rgba(168, 85, 247, 0.3);
|
||||||
|
}
|
||||||
|
|
||||||
|
.node-group-badge.section {
|
||||||
|
background: rgba(59, 130, 246, 0.1);
|
||||||
|
color: #3b82f6;
|
||||||
|
border-color: rgba(59, 130, 246, 0.3);
|
||||||
|
}
|
||||||
|
|
||||||
|
.node-group-badge.bridge {
|
||||||
|
background: rgba(236, 72, 153, 0.1);
|
||||||
|
color: #ec4899;
|
||||||
|
border-color: rgba(236, 72, 153, 0.3);
|
||||||
|
}
|
||||||
|
|
||||||
|
.node-group-badge.current {
|
||||||
|
background: rgba(16, 185, 129, 0.1);
|
||||||
|
color: #10b981;
|
||||||
|
border-color: rgba(16, 185, 129, 0.3);
|
||||||
|
}
|
||||||
|
|
||||||
|
.node-group-badge.concept {
|
||||||
|
background: rgba(0, 240, 255, 0.1);
|
||||||
|
color: #00f0ff;
|
||||||
|
border-color: rgba(0, 240, 255, 0.3);
|
||||||
|
}
|
||||||
|
|
||||||
|
.node-label {
|
||||||
|
font-family: var(--nexus-font-sans, "Outfit", sans-serif);
|
||||||
|
font-size: 1.15rem;
|
||||||
|
font-weight: 600;
|
||||||
|
color: #ffffff;
|
||||||
|
margin: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.detail-section {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 0.5rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.section-title {
|
||||||
|
font-size: 0.7rem;
|
||||||
|
font-weight: 600;
|
||||||
|
text-transform: uppercase;
|
||||||
|
letter-spacing: 0.08em;
|
||||||
|
margin: 0;
|
||||||
|
color: rgba(255, 255, 255, 0.4);
|
||||||
|
}
|
||||||
|
|
||||||
|
.neon-sub-header {
|
||||||
|
border-left: 2px solid var(--nexus-neon, #00f0ff);
|
||||||
|
padding-left: 0.5rem;
|
||||||
|
text-shadow: 0 0 10px rgba(0, 240, 255, 0.2);
|
||||||
|
}
|
||||||
|
|
||||||
|
.node-description {
|
||||||
|
font-size: 0.85rem;
|
||||||
|
line-height: 1.5;
|
||||||
|
color: rgba(255, 255, 255, 0.85);
|
||||||
|
margin: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.node-summary {
|
||||||
|
font-size: 0.82rem;
|
||||||
|
line-height: 1.5;
|
||||||
|
color: rgba(255, 255, 255, 0.75);
|
||||||
|
background: rgba(255, 255, 255, 0.02);
|
||||||
|
border-left: 2px solid rgba(255, 255, 255, 0.1);
|
||||||
|
padding: 0.5rem 0.75rem;
|
||||||
|
border-radius: 0 4px 4px 0;
|
||||||
|
margin: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.key-terms-list {
|
||||||
|
list-style: none;
|
||||||
|
padding: 0;
|
||||||
|
margin: 0;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 0.4rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.key-term-item {
|
||||||
|
display: flex;
|
||||||
|
align-items: flex-start;
|
||||||
|
gap: 0.5rem;
|
||||||
|
font-size: 0.82rem;
|
||||||
|
color: rgba(255, 255, 255, 0.8);
|
||||||
|
}
|
||||||
|
|
||||||
|
.term-bullet {
|
||||||
|
color: var(--nexus-neon, #00f0ff);
|
||||||
|
filter: drop-shadow(0 0 3px var(--nexus-neon, #00f0ff));
|
||||||
|
font-weight: bold;
|
||||||
|
}
|
||||||
|
|
||||||
|
.term-text {
|
||||||
|
line-height: 1.4;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Sidebar Footer & Open Quiz Button */
|
||||||
|
.sidebar-footer {
|
||||||
|
padding: 1rem 1.25rem;
|
||||||
|
background: rgba(13, 13, 13, 0.95);
|
||||||
|
border-top: 1px solid rgba(255, 255, 255, 0.05);
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 0.5rem;
|
||||||
|
z-index: 10;
|
||||||
|
}
|
||||||
|
|
||||||
|
.open-quiz-btn {
|
||||||
|
width: 100%;
|
||||||
|
padding: 0.75rem 1rem;
|
||||||
|
border-radius: 8px;
|
||||||
|
font-family: var(--nexus-font-sans, "Outfit", sans-serif);
|
||||||
|
font-size: 0.75rem;
|
||||||
|
font-weight: 700;
|
||||||
|
letter-spacing: 0.12em;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
gap: 0.6rem;
|
||||||
|
cursor: pointer;
|
||||||
|
transition: all 0.3s cubic-bezier(0.4, 0, 0.2, 1);
|
||||||
|
background: rgba(0, 240, 255, 0.03);
|
||||||
|
border: 1px solid rgba(0, 240, 255, 0.3);
|
||||||
|
color: var(--nexus-neon, #00f0ff);
|
||||||
|
box-shadow: 0 4px 15px rgba(0, 240, 255, 0.05);
|
||||||
|
}
|
||||||
|
|
||||||
|
.open-quiz-btn:hover {
|
||||||
|
background: rgba(0, 240, 255, 0.1);
|
||||||
|
border-color: var(--nexus-neon, #00f0ff);
|
||||||
|
color: #ffffff;
|
||||||
|
box-shadow: 0 0 20px rgba(0, 240, 255, 0.25);
|
||||||
|
transform: translateY(-2px);
|
||||||
|
}
|
||||||
|
|
||||||
|
.open-quiz-btn:active {
|
||||||
|
transform: translateY(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
.quiz-pulse-btn {
|
||||||
|
animation: quiz-pulse-glow 2s infinite ease-in-out;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes quiz-pulse-glow {
|
||||||
|
0% { border-color: rgba(0, 240, 255, 0.3); box-shadow: 0 0 5px rgba(0, 240, 255, 0.1); }
|
||||||
|
50% { border-color: var(--nexus-neon, #00f0ff); box-shadow: 0 0 25px rgba(0, 240, 255, 0.3); }
|
||||||
|
100% { border-color: rgba(0, 240, 255, 0.3); box-shadow: 0 0 5px rgba(0, 240, 255, 0.1); }
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Quiz Navigation Header */
|
||||||
|
.quiz-layout {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
}
|
||||||
|
|
||||||
|
.quiz-nav {
|
||||||
|
padding: 0.75rem 1.25rem;
|
||||||
|
border-bottom: 1px solid rgba(255, 255, 255, 0.05);
|
||||||
|
background: rgba(255, 255, 255, 0.01);
|
||||||
|
}
|
||||||
|
|
||||||
|
.back-to-graph-btn {
|
||||||
|
background: none;
|
||||||
|
border: none;
|
||||||
|
color: rgba(255, 255, 255, 0.5);
|
||||||
|
font-size: 0.8rem;
|
||||||
|
font-weight: 500;
|
||||||
|
cursor: pointer;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 0.5rem;
|
||||||
|
padding: 0.25rem 0.5rem;
|
||||||
|
border-radius: 4px;
|
||||||
|
transition: all 0.2s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.back-to-graph-btn:hover {
|
||||||
|
color: var(--nexus-neon, #00f0ff);
|
||||||
|
background: rgba(255, 255, 255, 0.02);
|
||||||
}
|
}
|
||||||
@@ -5,7 +5,10 @@
|
|||||||
@using NexusReader.UI.Shared.Services
|
@using NexusReader.UI.Shared.Services
|
||||||
@inject IIdentityService IdentityService
|
@inject IIdentityService IdentityService
|
||||||
@inject NavigationManager NavigationManager
|
@inject NavigationManager NavigationManager
|
||||||
|
@inject ISyncService SyncService
|
||||||
@attribute [Authorize]
|
@attribute [Authorize]
|
||||||
|
@implements IDisposable
|
||||||
|
|
||||||
|
|
||||||
<PageTitle>Dashboard | Nexus Reader</PageTitle>
|
<PageTitle>Dashboard | Nexus Reader</PageTitle>
|
||||||
|
|
||||||
@@ -18,20 +21,20 @@
|
|||||||
<img src="https://api.dicebear.com/7.x/bottts/svg?seed=Nexus" alt="Profile" class="profile-img" />
|
<img src="https://api.dicebear.com/7.x/bottts/svg?seed=Nexus" alt="Profile" class="profile-img" />
|
||||||
<div class="avatar-glow"></div>
|
<div class="avatar-glow"></div>
|
||||||
</div>
|
</div>
|
||||||
<h1 class="username">[User_Explorer1988]</h1>
|
<h1 class="username">@(string.IsNullOrEmpty(_profile?.DisplayName) ? (_profile?.Email.Split('@')[0] ?? "Użytkownik") : _profile.DisplayName)</h1>
|
||||||
|
|
||||||
<div class="status-pills">
|
<div class="status-pills">
|
||||||
<div class="status-pill">
|
<div class="status-pill">
|
||||||
<span class="pill-label">Books Read:</span>
|
<span class="pill-label">Książki:</span>
|
||||||
<span class="pill-value">12</span>
|
<span class="pill-value">@(_profile?.BooksReadCount ?? 0)</span>
|
||||||
</div>
|
</div>
|
||||||
<div class="status-pill">
|
<div class="status-pill">
|
||||||
<span class="pill-label">Concepts Mapped:</span>
|
<span class="pill-label">Pojęcia:</span>
|
||||||
<span class="pill-value">450</span>
|
<span class="pill-value">@(_profile?.ConceptsMappedCount ?? 0)</span>
|
||||||
</div>
|
</div>
|
||||||
<div class="status-pill">
|
<div class="status-pill">
|
||||||
<span class="pill-label">Quiz Mastery:</span>
|
<span class="pill-label">Średni Wynik:</span>
|
||||||
<span class="pill-value">88%</span>
|
<span class="pill-value">@(_profile?.AverageQuizScore ?? 0)%</span>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -39,7 +42,7 @@
|
|||||||
|
|
||||||
<!-- Main Content Area -->
|
<!-- Main Content Area -->
|
||||||
<main class="dashboard-content">
|
<main class="dashboard-content">
|
||||||
<h2 class="section-title">Witaj, @(_profile?.Email.Split('@')[0] ?? "Użytkowniku")</h2>
|
<h2 class="section-title">Witaj, @(string.IsNullOrEmpty(_profile?.DisplayName) ? (_profile?.Email.Split('@')[0] ?? "Użytkowniku") : _profile.DisplayName)</h2>
|
||||||
|
|
||||||
<div class="main-grid">
|
<div class="main-grid">
|
||||||
<!-- Current Reading Card -->
|
<!-- Current Reading Card -->
|
||||||
@@ -49,34 +52,88 @@
|
|||||||
<!-- Knowledge Integration -->
|
<!-- Knowledge Integration -->
|
||||||
<section class="integration-card glass-panel">
|
<section class="integration-card glass-panel">
|
||||||
<div class="panel-header">
|
<div class="panel-header">
|
||||||
<h4>Knowledge Integration Progress</h4>
|
<h4>Integracja Wiedzy</h4>
|
||||||
<NexusIcon Name="arrow-right" Size="16" />
|
<NexusIcon Name="arrow-right" Size="16" />
|
||||||
</div>
|
</div>
|
||||||
<div class="graph-placeholder">
|
<div class="graph-placeholder">
|
||||||
<div class="graph-node central"></div>
|
<div class="graph-node central" title="Ośrodek Wiedzy Nexus Reader"></div>
|
||||||
<div class="graph-node satellite" style="--angle: 0deg; --dist: 60px;"></div>
|
|
||||||
<div class="graph-node satellite" style="--angle: 120deg; --dist: 50px;"></div>
|
@if (_profile?.MappedConcepts != null && _profile.MappedConcepts.Any())
|
||||||
<div class="graph-node satellite" style="--angle: 240deg; --dist: 70px;"></div>
|
{
|
||||||
<div class="active-node-label">TU JESTEŚ</div>
|
@for (int i = 0; i < _profile.MappedConcepts.Count; i++)
|
||||||
|
{
|
||||||
|
var concept = _profile.MappedConcepts[i];
|
||||||
|
var angle = i * (360.0 / _profile.MappedConcepts.Count);
|
||||||
|
var dist = 65;
|
||||||
|
<div class="graph-node satellite"
|
||||||
|
style="--angle: @(angle)deg; --dist: @(dist)px;"
|
||||||
|
title="[@concept.Type] @concept.Content"
|
||||||
|
@onmouseover="() => SetHoveredConcept(concept)"
|
||||||
|
@onmouseout="ClearHoveredConcept">
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
<div class="graph-node satellite" style="--angle: 0deg; --dist: 60px;"></div>
|
||||||
|
<div class="graph-node satellite" style="--angle: 120deg; --dist: 50px;"></div>
|
||||||
|
<div class="graph-node satellite" style="--angle: 240deg; --dist: 70px;"></div>
|
||||||
|
}
|
||||||
|
|
||||||
|
<div class="active-node-label">
|
||||||
|
@(string.IsNullOrEmpty(_hoveredConceptLabel) ? "TU JESTEŚ" : _hoveredConceptLabel)
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
@if (_hoveredConcept != null)
|
||||||
|
{
|
||||||
|
<div class="concept-detail-toast">
|
||||||
|
<span class="concept-type">@_hoveredConcept.Type</span>
|
||||||
|
<p class="concept-content">@_hoveredConcept.Content</p>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
<div class="concept-detail-toast placeholder">
|
||||||
|
<span class="concept-type">Mapowanie AI</span>
|
||||||
|
<p class="concept-content">Najedź na węzeł, aby zbadać pojęcie wydobyte przez Nexus AI.</p>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
</section>
|
</section>
|
||||||
|
|
||||||
<!-- Quiz Summary -->
|
<!-- Quiz Summary -->
|
||||||
<section class="quiz-card glass-panel">
|
<section class="quiz-card glass-panel">
|
||||||
<div class="panel-header">
|
<div class="panel-header">
|
||||||
<h4>Quiz Summary: Key Thinkers</h4>
|
<h4>Rozwiązane Quizy</h4>
|
||||||
<NexusIcon Name="arrow-right" Size="16" />
|
<NexusIcon Name="arrow-right" Size="16" />
|
||||||
</div>
|
</div>
|
||||||
<div class="quiz-preview">
|
<div class="quiz-preview">
|
||||||
<p class="question">Który artysta namalował 'Ostatnią Wieczerzę'?</p>
|
@if (_profile?.RecentQuizzes != null && _profile.RecentQuizzes.Any())
|
||||||
<div class="quiz-options">
|
{
|
||||||
<div class="quiz-option active">
|
<div class="quiz-history-list">
|
||||||
<span class="option-letter">A)</span> Michal Anioł
|
@foreach (var quiz in _profile.RecentQuizzes)
|
||||||
|
{
|
||||||
|
<div class="quiz-history-item">
|
||||||
|
<div class="quiz-item-header">
|
||||||
|
<span class="quiz-topic">@quiz.Topic</span>
|
||||||
|
<span class="quiz-score badge @(quiz.Percentage >= 80 ? "badge-success" : quiz.Percentage >= 50 ? "badge-warning" : "badge-danger")">
|
||||||
|
@quiz.Score / @quiz.TotalQuestions (@((int)quiz.Percentage)%)
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div class="quiz-item-meta">
|
||||||
|
<span class="quiz-date">@quiz.CompletedDate.ToString("g")</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
</div>
|
</div>
|
||||||
<div class="quiz-option">
|
}
|
||||||
<span class="option-letter">B)</span> Leonardo da Vinci
|
else
|
||||||
|
{
|
||||||
|
<div class="empty-quiz-state">
|
||||||
|
<p class="question">Brak rozwiązanych quizów</p>
|
||||||
|
<p class="sub-text">Rozwiązuj quizy w trakcie czytania książek, aby śledzić swoje postępy.</p>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
}
|
||||||
</div>
|
</div>
|
||||||
</section>
|
</section>
|
||||||
</div>
|
</div>
|
||||||
@@ -86,13 +143,65 @@
|
|||||||
|
|
||||||
@code {
|
@code {
|
||||||
private UserProfileDto? _profile;
|
private UserProfileDto? _profile;
|
||||||
|
private MappedConceptDto? _hoveredConcept;
|
||||||
|
private string _hoveredConceptLabel = string.Empty;
|
||||||
|
|
||||||
protected override async Task OnInitializedAsync()
|
protected override async Task OnInitializedAsync()
|
||||||
|
{
|
||||||
|
IdentityService.OnStateInvalidated += HandleStateInvalidatedAsync;
|
||||||
|
await LoadProfileAsync();
|
||||||
|
|
||||||
|
await SyncService.InitializeAsync();
|
||||||
|
SyncService.OnProgressReceived += HandleProgressReceivedAsync;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void SetHoveredConcept(MappedConceptDto concept)
|
||||||
|
{
|
||||||
|
_hoveredConcept = concept;
|
||||||
|
_hoveredConceptLabel = concept.DisplayLabel;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void ClearHoveredConcept()
|
||||||
|
{
|
||||||
|
_hoveredConcept = null;
|
||||||
|
_hoveredConceptLabel = string.Empty;
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task LoadProfileAsync()
|
||||||
{
|
{
|
||||||
var result = await IdentityService.GetProfileAsync();
|
var result = await IdentityService.GetProfileAsync();
|
||||||
if (result.IsSuccess)
|
if (result.IsSuccess)
|
||||||
{
|
{
|
||||||
_profile = result.Value;
|
_profile = result.Value;
|
||||||
}
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
_profile = null;
|
||||||
|
}
|
||||||
|
StateHasChanged();
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task HandleStateInvalidatedAsync()
|
||||||
|
{
|
||||||
|
await InvokeAsync(async () =>
|
||||||
|
{
|
||||||
|
await LoadProfileAsync();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task HandleProgressReceivedAsync(string pageId, DateTime timestamp)
|
||||||
|
{
|
||||||
|
await InvokeAsync(async () =>
|
||||||
|
{
|
||||||
|
IdentityService.ClearCache();
|
||||||
|
await LoadProfileAsync();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Dispose()
|
||||||
|
{
|
||||||
|
IdentityService.OnStateInvalidated -= HandleStateInvalidatedAsync;
|
||||||
|
SyncService.OnProgressReceived -= HandleProgressReceivedAsync;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -294,9 +294,19 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
.graph-node.satellite {
|
.graph-node.satellite {
|
||||||
width: 20px;
|
width: 16px;
|
||||||
height: 20px;
|
height: 16px;
|
||||||
transform: rotate(var(--angle)) translateY(var(--dist));
|
transform: rotate(var(--angle)) translateY(var(--dist));
|
||||||
|
background: rgba(0, 255, 153, 0.4);
|
||||||
|
border: 1px solid var(--nexus-neon);
|
||||||
|
cursor: pointer;
|
||||||
|
transition: all 0.3s cubic-bezier(0.4, 0, 0.2, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
.graph-node.satellite:hover {
|
||||||
|
background: var(--nexus-neon);
|
||||||
|
box-shadow: 0 0 15px var(--nexus-neon);
|
||||||
|
transform: rotate(var(--angle)) translateY(var(--dist)) scale(1.3);
|
||||||
}
|
}
|
||||||
|
|
||||||
.active-node-label {
|
.active-node-label {
|
||||||
@@ -404,3 +414,117 @@
|
|||||||
grid-template-columns: 1fr;
|
grid-template-columns: 1fr;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* --- Quiz History Styling --- */
|
||||||
|
.quiz-history-list {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 1rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.quiz-history-item {
|
||||||
|
background: rgba(255, 255, 255, 0.02);
|
||||||
|
border: 1px solid rgba(255, 255, 255, 0.05);
|
||||||
|
border-radius: 12px;
|
||||||
|
padding: 1rem;
|
||||||
|
transition: all 0.2s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
.quiz-history-item:hover {
|
||||||
|
background: rgba(255, 255, 255, 0.04);
|
||||||
|
border-color: rgba(255, 255, 255, 0.1);
|
||||||
|
}
|
||||||
|
|
||||||
|
.quiz-item-header {
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
align-items: center;
|
||||||
|
gap: 1rem;
|
||||||
|
margin-bottom: 0.5rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.quiz-topic {
|
||||||
|
font-size: 0.95rem;
|
||||||
|
font-weight: 500;
|
||||||
|
color: #ffffff;
|
||||||
|
}
|
||||||
|
|
||||||
|
.quiz-item-meta {
|
||||||
|
display: flex;
|
||||||
|
font-size: 0.75rem;
|
||||||
|
color: #666666;
|
||||||
|
}
|
||||||
|
|
||||||
|
.badge {
|
||||||
|
padding: 0.25rem 0.5rem;
|
||||||
|
border-radius: 6px;
|
||||||
|
font-size: 0.75rem;
|
||||||
|
font-weight: 600;
|
||||||
|
}
|
||||||
|
|
||||||
|
.badge-success {
|
||||||
|
background: rgba(0, 255, 153, 0.1);
|
||||||
|
color: var(--nexus-neon);
|
||||||
|
border: 1px solid rgba(0, 255, 153, 0.3);
|
||||||
|
}
|
||||||
|
|
||||||
|
.badge-warning {
|
||||||
|
background: rgba(255, 170, 0, 0.1);
|
||||||
|
color: #ffa800;
|
||||||
|
border: 1px solid rgba(255, 170, 0, 0.3);
|
||||||
|
}
|
||||||
|
|
||||||
|
.badge-danger {
|
||||||
|
background: rgba(255, 50, 50, 0.1);
|
||||||
|
color: #ff3232;
|
||||||
|
border: 1px solid rgba(255, 50, 50, 0.3);
|
||||||
|
}
|
||||||
|
|
||||||
|
.empty-quiz-state {
|
||||||
|
text-align: center;
|
||||||
|
padding: 2rem 1rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.empty-quiz-state .sub-text {
|
||||||
|
font-size: 0.8rem;
|
||||||
|
color: #666666;
|
||||||
|
margin-top: 0.5rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* --- Concept Detail Toast for Dashboard --- */
|
||||||
|
.concept-detail-toast {
|
||||||
|
margin-top: 1rem;
|
||||||
|
padding: 0.75rem 1rem;
|
||||||
|
background: rgba(255, 255, 255, 0.02);
|
||||||
|
border: 1px solid rgba(255, 255, 255, 0.05);
|
||||||
|
border-radius: 12px;
|
||||||
|
min-height: 80px;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 0.25rem;
|
||||||
|
transition: all 0.3s cubic-bezier(0.4, 0, 0.2, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
.concept-detail-toast.placeholder {
|
||||||
|
opacity: 0.5;
|
||||||
|
}
|
||||||
|
|
||||||
|
.concept-type {
|
||||||
|
font-size: 0.75rem;
|
||||||
|
font-weight: 700;
|
||||||
|
color: var(--nexus-neon);
|
||||||
|
text-transform: uppercase;
|
||||||
|
letter-spacing: 1px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.concept-content {
|
||||||
|
font-size: 0.85rem;
|
||||||
|
line-height: 1.4;
|
||||||
|
color: #E0E0E0;
|
||||||
|
margin: 0;
|
||||||
|
display: -webkit-box;
|
||||||
|
-webkit-line-clamp: 2;
|
||||||
|
-webkit-box-orient: vertical;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -3,179 +3,430 @@
|
|||||||
@using NexusReader.Application.DTOs.AI
|
@using NexusReader.Application.DTOs.AI
|
||||||
@using NexusReader.Application.Abstractions.Services
|
@using NexusReader.Application.Abstractions.Services
|
||||||
@using NexusReader.Application.DTOs.User
|
@using NexusReader.Application.DTOs.User
|
||||||
|
@using NexusReader.UI.Shared.Components.Atoms
|
||||||
@using System.Net.Http.Json
|
@using System.Net.Http.Json
|
||||||
@inject HttpClient Http
|
@inject HttpClient Http
|
||||||
@inject IKnowledgeService KnowledgeService
|
@inject IKnowledgeService KnowledgeService
|
||||||
|
@inject AuthenticationStateProvider AuthStateProvider
|
||||||
|
|
||||||
<div class="intelligence-page">
|
<div class="intelligence-page">
|
||||||
<header class="intelligence-header">
|
<header class="intelligence-header">
|
||||||
<div class="header-title-section">
|
<div class="header-title-section">
|
||||||
<h1>Global AI Q&A</h1>
|
<h1 class="neon-glow-text">Global Intelligence</h1>
|
||||||
<p class="subtitle">Search, interrogate, and extract grounded facts from your library using Polyglot KM-RAG</p>
|
<p class="subtitle">Interrogate, explore, and synthesize grounded knowledge from your library using Polyglot KM-RAG</p>
|
||||||
</div>
|
</div>
|
||||||
</header>
|
</header>
|
||||||
|
|
||||||
<div class="intelligence-layout glass-panel">
|
<div class="intelligence-layout glass-panel">
|
||||||
<div class="search-scope-bar">
|
<div class="chat-thread-container">
|
||||||
<div class="input-group search-input-group">
|
@if (_chatMessages.Count == 0)
|
||||||
<input class="nexus-input"
|
|
||||||
placeholder="Ask a question about your books..."
|
|
||||||
@bind="_question"
|
|
||||||
@bind:event="oninput"
|
|
||||||
@onkeyup="HandleKeyUp" />
|
|
||||||
<button class="btn-nexus primary search-btn"
|
|
||||||
disabled="@(string.IsNullOrWhiteSpace(_question) || _isLoading)"
|
|
||||||
@onclick="AskQuestionAsync">
|
|
||||||
@if (_isLoading)
|
|
||||||
{
|
|
||||||
<div class="spinner-glow small btn-spinner"></div>
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
<span>Ask AI</span>
|
|
||||||
}
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="scope-selector">
|
|
||||||
<label for="book-select">Scope:</label>
|
|
||||||
<select id="book-select" class="nexus-select" @bind="_selectedBookId">
|
|
||||||
<option value="">All Books (Global Search)</option>
|
|
||||||
@if (_books != null)
|
|
||||||
{
|
|
||||||
@foreach (var book in _books)
|
|
||||||
{
|
|
||||||
<option value="@book.Id">@book.Title</option>
|
|
||||||
}
|
|
||||||
}
|
|
||||||
</select>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="results-area">
|
|
||||||
@if (_isLoading)
|
|
||||||
{
|
{
|
||||||
<div class="loading-state">
|
<div class="welcome-state">
|
||||||
<div class="nexus-spinner"></div>
|
<div class="welcome-icon">
|
||||||
<span>Analyzing conceptual graph and synthesizing response...</span>
|
<svg width="64" height="64" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.5" stroke-linecap="round" stroke-linejoin="round">
|
||||||
|
<path d="M21 15a2 2 0 0 1-2 2H7l-4 4V5a2 2 0 0 1 2-2h14a2 2 0 0 1 2 2z"></path>
|
||||||
|
</svg>
|
||||||
|
</div>
|
||||||
|
<h3>Start Interrogating Your Library</h3>
|
||||||
|
<p>Ask complex questions across your entire ebook collection. The KM-RAG engine dynamically builds semantic maps, resolves dependencies, and formulates high-fidelity, grounded answers with interactive popover citations.</p>
|
||||||
</div>
|
</div>
|
||||||
}
|
}
|
||||||
else if (_response != null)
|
else
|
||||||
{
|
{
|
||||||
<div class="response-container">
|
<div class="chat-bubbles-scroll">
|
||||||
<div class="response-section">
|
@foreach (var message in _chatMessages)
|
||||||
<h4><i class="bi bi-robot"></i> Answer</h4>
|
|
||||||
<div class="answer-text">
|
|
||||||
@_response.Answer
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
@if (_response.Citations != null && _response.Citations.Any())
|
|
||||||
{
|
{
|
||||||
<div class="citations-section">
|
<div class="message-row @(message.Sender == "User" ? "user-row" : "ai-row")" key="@message.Id">
|
||||||
<h4><i class="bi bi-journal-check"></i> Grounded Citations</h4>
|
<div class="message-avatar">
|
||||||
<div class="citations-grid">
|
@if (message.Sender == "User")
|
||||||
@foreach (var citation in _response.Citations)
|
|
||||||
{
|
{
|
||||||
<div class="citation-card">
|
<i class="bi bi-person-fill"></i>
|
||||||
<div class="citation-header">
|
|
||||||
<span class="source-badge">@citation.SourceBook</span>
|
|
||||||
@if (!string.IsNullOrEmpty(citation.CitationId) && citation.CitationId.Length > 8)
|
|
||||||
{
|
|
||||||
<span class="id-badge">ID: @citation.CitationId.Substring(0, Math.Min(8, citation.CitationId.Length))</span>
|
|
||||||
}
|
|
||||||
</div>
|
|
||||||
<div class="citation-body">
|
|
||||||
"@citation.Snippet"
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
}
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
<i class="bi bi-robot"></i>
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
<div class="message-bubble @(message.Sender == "User" ? "user-bubble" : "ai-bubble")">
|
||||||
|
<div class="message-header">
|
||||||
|
<span class="sender-name">@message.Sender</span>
|
||||||
|
<span class="message-time">@message.Timestamp.ToString("HH:mm")</span>
|
||||||
|
</div>
|
||||||
|
<div class="message-content">
|
||||||
|
@foreach (var segment in message.Segments)
|
||||||
|
{
|
||||||
|
@if (segment.IsCitation)
|
||||||
|
{
|
||||||
|
<NexusCitationMarker SourceId="@segment.CitationId" Citations="@message.Citations" />
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
@RenderMarkdown(segment.Text)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
|
||||||
|
@if (_isLoading)
|
||||||
|
{
|
||||||
|
<div class="message-row ai-row">
|
||||||
|
<div class="message-avatar">
|
||||||
|
<i class="bi bi-robot"></i>
|
||||||
|
</div>
|
||||||
|
<div class="message-bubble ai-bubble pending-bubble">
|
||||||
|
<div class="message-header">
|
||||||
|
<span class="sender-name">AI</span>
|
||||||
|
<span class="message-time">Thinking...</span>
|
||||||
|
</div>
|
||||||
|
<div class="message-content">
|
||||||
|
<div class="typing-indicator">
|
||||||
|
<span></span>
|
||||||
|
<span></span>
|
||||||
|
<span></span>
|
||||||
|
</div>
|
||||||
|
<span class="loading-label">Analyzing conceptual graphs and synthesizing response...</span>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
}
|
}
|
||||||
</div>
|
</div>
|
||||||
}
|
}
|
||||||
else if (_hasSearched)
|
</div>
|
||||||
{
|
|
||||||
<div class="empty-state">
|
<div class="chat-input-controls">
|
||||||
<i class="bi bi-info-circle"></i>
|
<div class="input-panel-wrapper">
|
||||||
<p>No answers generated. Try adjusting your question.</p>
|
<div class="scope-bar">
|
||||||
</div>
|
<div class="scope-selector">
|
||||||
}
|
<label for="book-select"><i class="bi bi-compass"></i> Scope:</label>
|
||||||
else
|
<select id="book-select" class="nexus-select" @bind="_selectedBookId">
|
||||||
{
|
<option value="">All Books (Global Search)</option>
|
||||||
<div class="welcome-state">
|
@if (_books != null)
|
||||||
<div class="welcome-icon">
|
{
|
||||||
<svg width="48" height="48" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.5" stroke-linecap="round" stroke-linejoin="round">
|
@foreach (var book in _books)
|
||||||
<path d="M21 15a2 2 0 0 1-2 2H7l-4 4V5a2 2 0 0 1 2-2h14a2 2 0 0 1 2 2z"></path>
|
{
|
||||||
</svg>
|
<option value="@book.Id">@book.Title</option>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</select>
|
||||||
</div>
|
</div>
|
||||||
<h3>Start Interrogating Your Library</h3>
|
|
||||||
<p>Ask complex questions across all your books. The system will search vectors, pull concept graph relations, and formulate a grounded answer with precise citations.</p>
|
|
||||||
</div>
|
</div>
|
||||||
}
|
|
||||||
|
<div class="input-field-group">
|
||||||
|
<input class="nexus-input"
|
||||||
|
placeholder="Ask a question about your books..."
|
||||||
|
@bind="_question"
|
||||||
|
@bind:event="oninput"
|
||||||
|
@onkeyup="HandleKeyUp"
|
||||||
|
disabled="@_isLoading" />
|
||||||
|
<button class="btn-nexus primary search-btn"
|
||||||
|
disabled="@(string.IsNullOrWhiteSpace(_question) || _isLoading)"
|
||||||
|
@onclick="AskQuestionAsync">
|
||||||
|
@if (_isLoading)
|
||||||
|
{
|
||||||
|
<div class="btn-spinner"></div>
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
<span><i class="bi bi-send-fill"></i></span>
|
||||||
|
}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<style>
|
<style>
|
||||||
.intelligence-page {
|
.intelligence-page {
|
||||||
padding: 3rem 2rem;
|
padding: 2rem;
|
||||||
max-width: 1100px;
|
max-width: 1100px;
|
||||||
margin: 0 auto;
|
margin: 0 auto;
|
||||||
|
height: calc(100vh - 100px);
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
animation: fadeIn 0.5s ease-out;
|
animation: fadeIn 0.5s ease-out;
|
||||||
}
|
}
|
||||||
|
|
||||||
.intelligence-header {
|
.intelligence-header {
|
||||||
margin-bottom: 2rem;
|
margin-bottom: 1.5rem;
|
||||||
|
flex-shrink: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
.header-title-section h1 {
|
.neon-glow-text {
|
||||||
font-family: var(--nexus-font-serif, 'Outfit', 'Georgia', serif);
|
font-family: var(--nexus-font-sans, 'Outfit', sans-serif);
|
||||||
font-size: 2.8rem;
|
font-size: 2.5rem;
|
||||||
font-weight: 700;
|
font-weight: 800;
|
||||||
margin: 0 0 0.5rem 0;
|
margin: 0 0 0.25rem 0;
|
||||||
background: linear-gradient(135deg, var(--nexus-text, #ffffff) 0%, rgba(255, 255, 255, 0.7) 100%);
|
background: linear-gradient(135deg, #00ff99 0%, #06b6d4 100%);
|
||||||
-webkit-background-clip: text;
|
-webkit-background-clip: text;
|
||||||
-webkit-text-fill-color: transparent;
|
-webkit-text-fill-color: transparent;
|
||||||
|
filter: drop-shadow(0 0 8px rgba(6, 182, 212, 0.2));
|
||||||
}
|
}
|
||||||
|
|
||||||
.subtitle {
|
.subtitle {
|
||||||
font-size: 1rem;
|
font-size: 0.95rem;
|
||||||
color: rgba(255, 255, 255, 0.6);
|
color: rgba(255, 255, 255, 0.6);
|
||||||
margin: 0;
|
margin: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
.intelligence-layout {
|
.intelligence-layout {
|
||||||
padding: 2.5rem;
|
|
||||||
border-radius: var(--nexus-radius-lg, 16px);
|
|
||||||
background: rgba(15, 23, 42, 0.4);
|
|
||||||
border: 1px solid rgba(255, 255, 255, 0.08);
|
|
||||||
backdrop-filter: blur(20px);
|
|
||||||
box-shadow: 0 20px 50px rgba(0, 0, 0, 0.3);
|
|
||||||
}
|
|
||||||
|
|
||||||
.search-scope-bar {
|
|
||||||
display: flex;
|
|
||||||
gap: 1.5rem;
|
|
||||||
align-items: center;
|
|
||||||
margin-bottom: 2.5rem;
|
|
||||||
flex-wrap: wrap;
|
|
||||||
}
|
|
||||||
|
|
||||||
.search-input-group {
|
|
||||||
flex-grow: 1;
|
flex-grow: 1;
|
||||||
display: flex;
|
display: flex;
|
||||||
background: rgba(15, 23, 42, 0.5);
|
flex-direction: column;
|
||||||
border: 1px solid rgba(255, 255, 255, 0.1);
|
border-radius: 20px;
|
||||||
border-radius: 30px;
|
background: rgba(10, 16, 26, 0.45);
|
||||||
padding: 0.25rem 0.25rem 0.25rem 1.25rem;
|
border: 1px solid rgba(6, 182, 212, 0.15);
|
||||||
transition: all 0.3s ease;
|
backdrop-filter: blur(16px);
|
||||||
|
-webkit-backdrop-filter: blur(16px);
|
||||||
|
box-shadow: 0 15px 35px rgba(0, 0, 0, 0.4), 0 0 20px rgba(6, 182, 212, 0.05);
|
||||||
|
overflow: hidden;
|
||||||
}
|
}
|
||||||
|
|
||||||
.search-input-group:focus-within {
|
.chat-thread-container {
|
||||||
border-color: var(--nexus-primary, #6366f1);
|
flex-grow: 1;
|
||||||
box-shadow: 0 0 10px rgba(99, 102, 241, 0.2);
|
overflow-y: auto;
|
||||||
|
padding: 2rem;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Custom Scrollbars */
|
||||||
|
.chat-thread-container::-webkit-scrollbar {
|
||||||
|
width: 6px;
|
||||||
|
}
|
||||||
|
.chat-thread-container::-webkit-scrollbar-track {
|
||||||
|
background: rgba(255, 255, 255, 0.01);
|
||||||
|
}
|
||||||
|
.chat-thread-container::-webkit-scrollbar-thumb {
|
||||||
|
background: rgba(6, 182, 212, 0.2);
|
||||||
|
border-radius: 4px;
|
||||||
|
}
|
||||||
|
.chat-thread-container::-webkit-scrollbar-thumb:hover {
|
||||||
|
background: rgba(6, 182, 212, 0.4);
|
||||||
|
}
|
||||||
|
|
||||||
|
.chat-bubbles-scroll {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 1.5rem;
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-row {
|
||||||
|
display: flex;
|
||||||
|
gap: 1rem;
|
||||||
|
width: 100%;
|
||||||
|
max-width: 85%;
|
||||||
|
animation: bubble-fade-in 0.3s cubic-bezier(0.16, 1, 0.3, 1) forwards;
|
||||||
|
}
|
||||||
|
|
||||||
|
.user-row {
|
||||||
|
align-self: flex-end;
|
||||||
|
flex-direction: row-reverse;
|
||||||
|
}
|
||||||
|
|
||||||
|
.ai-row {
|
||||||
|
align-self: flex-start;
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-avatar {
|
||||||
|
width: 38px;
|
||||||
|
height: 38px;
|
||||||
|
border-radius: 50%;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
font-size: 1.1rem;
|
||||||
|
flex-shrink: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.user-row .message-avatar {
|
||||||
|
background: linear-gradient(135deg, #7c3aed 0%, #4c1d95 100%);
|
||||||
|
color: #f5f3ff;
|
||||||
|
border: 1px solid rgba(139, 92, 246, 0.4);
|
||||||
|
box-shadow: 0 0 10px rgba(139, 92, 246, 0.25);
|
||||||
|
}
|
||||||
|
|
||||||
|
.ai-row .message-avatar {
|
||||||
|
background: linear-gradient(135deg, #0f766e 0%, #115e59 100%);
|
||||||
|
color: #ccfbf1;
|
||||||
|
border: 1px solid rgba(13, 148, 136, 0.4);
|
||||||
|
box-shadow: 0 0 10px rgba(13, 148, 136, 0.25);
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-bubble {
|
||||||
|
padding: 1.25rem 1.5rem;
|
||||||
|
border-radius: 16px;
|
||||||
|
position: relative;
|
||||||
|
line-height: 1.6;
|
||||||
|
font-size: 0.975rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.user-bubble {
|
||||||
|
background: rgba(43, 24, 80, 0.35);
|
||||||
|
border: 1px solid rgba(139, 92, 246, 0.25);
|
||||||
|
color: #f3e8ff;
|
||||||
|
border-top-right-radius: 4px;
|
||||||
|
box-shadow: 0 4px 15px rgba(139, 92, 246, 0.05);
|
||||||
|
}
|
||||||
|
|
||||||
|
.ai-bubble {
|
||||||
|
background: rgba(10, 20, 30, 0.55);
|
||||||
|
border: 1px solid rgba(6, 182, 212, 0.2);
|
||||||
|
color: #e2e8f0;
|
||||||
|
border-top-left-radius: 4px;
|
||||||
|
box-shadow: 0 4px 15px rgba(6, 182, 212, 0.05);
|
||||||
|
flex-grow: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-header {
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
align-items: center;
|
||||||
|
margin-bottom: 0.5rem;
|
||||||
|
font-size: 0.75rem;
|
||||||
|
opacity: 0.6;
|
||||||
|
}
|
||||||
|
|
||||||
|
.sender-name {
|
||||||
|
font-weight: 700;
|
||||||
|
text-transform: uppercase;
|
||||||
|
letter-spacing: 0.5px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-time {
|
||||||
|
font-family: monospace;
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-content {
|
||||||
|
word-break: break-word;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Paragraph Spacing & Markdown */
|
||||||
|
.message-content p {
|
||||||
|
margin: 0 0 1rem 0;
|
||||||
|
}
|
||||||
|
.message-content p:last-child {
|
||||||
|
margin-bottom: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.nexus-code-block {
|
||||||
|
background: rgba(0, 0, 0, 0.4) !important;
|
||||||
|
border: 1px solid rgba(255, 255, 255, 0.08) !important;
|
||||||
|
border-radius: 8px;
|
||||||
|
padding: 1rem;
|
||||||
|
margin: 1rem 0;
|
||||||
|
overflow-x: auto;
|
||||||
|
font-family: 'Fira Code', monospace;
|
||||||
|
font-size: 0.85rem;
|
||||||
|
color: #a7f3d0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.nexus-inline-code {
|
||||||
|
background: rgba(255, 255, 255, 0.06);
|
||||||
|
border: 1px solid rgba(255, 255, 255, 0.1);
|
||||||
|
border-radius: 4px;
|
||||||
|
padding: 0.15rem 0.35rem;
|
||||||
|
font-family: monospace;
|
||||||
|
font-size: 0.9em;
|
||||||
|
color: #f472b6; /* Light pink for inline code */
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Pending State Bubble */
|
||||||
|
.pending-bubble {
|
||||||
|
border-color: rgba(6, 182, 212, 0.4);
|
||||||
|
box-shadow: 0 0 15px rgba(6, 182, 212, 0.1);
|
||||||
|
}
|
||||||
|
|
||||||
|
.typing-indicator {
|
||||||
|
display: flex;
|
||||||
|
gap: 4px;
|
||||||
|
align-items: center;
|
||||||
|
margin-bottom: 0.5rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.typing-indicator span {
|
||||||
|
width: 8px;
|
||||||
|
height: 8px;
|
||||||
|
background: #06b6d4;
|
||||||
|
border-radius: 50%;
|
||||||
|
display: inline-block;
|
||||||
|
animation: typing-bounce 1.4s infinite ease-in-out both;
|
||||||
|
}
|
||||||
|
|
||||||
|
.typing-indicator span:nth-child(1) { animation-delay: -0.32s; }
|
||||||
|
.typing-indicator span:nth-child(2) { animation-delay: -0.16s; }
|
||||||
|
|
||||||
|
.loading-label {
|
||||||
|
font-size: 0.85rem;
|
||||||
|
color: rgba(255, 255, 255, 0.5);
|
||||||
|
font-style: italic;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Input Controls */
|
||||||
|
.chat-input-controls {
|
||||||
|
padding: 1.5rem 2rem 2rem 2rem;
|
||||||
|
background: rgba(0, 0, 0, 0.2);
|
||||||
|
border-top: 1px solid rgba(255, 255, 255, 0.05);
|
||||||
|
flex-shrink: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.input-panel-wrapper {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 0.75rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.scope-bar {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.scope-selector {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 0.5rem;
|
||||||
|
font-size: 0.85rem;
|
||||||
|
color: rgba(255, 255, 255, 0.5);
|
||||||
|
}
|
||||||
|
|
||||||
|
.nexus-select {
|
||||||
|
background: rgba(255, 255, 255, 0.02);
|
||||||
|
border: 1px solid rgba(255, 255, 255, 0.08);
|
||||||
|
color: #ffffff;
|
||||||
|
padding: 0.35rem 2rem 0.35rem 0.75rem;
|
||||||
|
border-radius: 8px;
|
||||||
|
outline: none;
|
||||||
|
cursor: pointer;
|
||||||
|
font-size: 0.85rem;
|
||||||
|
transition: all 0.3s ease;
|
||||||
|
appearance: none;
|
||||||
|
background-image: url("data:image/svg+xml;charset=UTF-8,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24' fill='none' stroke='white' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3e%3cpolyline points='6 9 12 15 18 9'%3e%3c/polyline%3e%3c/svg%3e");
|
||||||
|
background-repeat: no-repeat;
|
||||||
|
background-position: right 0.75rem center;
|
||||||
|
background-size: 0.85em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.nexus-select:focus {
|
||||||
|
border-color: #06b6d4;
|
||||||
|
box-shadow: 0 0 8px rgba(6, 182, 212, 0.2);
|
||||||
|
}
|
||||||
|
|
||||||
|
.input-field-group {
|
||||||
|
display: flex;
|
||||||
|
background: rgba(255, 255, 255, 0.02);
|
||||||
|
border: 1px solid rgba(255, 255, 255, 0.08);
|
||||||
|
border-radius: 12px;
|
||||||
|
padding: 0.35rem;
|
||||||
|
transition: all 0.3s cubic-bezier(0.4, 0, 0.2, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
.input-field-group:focus-within {
|
||||||
|
border-color: #06b6d4;
|
||||||
|
background: rgba(6, 182, 212, 0.01);
|
||||||
|
box-shadow: 0 0 15px rgba(6, 182, 212, 0.15);
|
||||||
}
|
}
|
||||||
|
|
||||||
.nexus-input {
|
.nexus-input {
|
||||||
@@ -185,205 +436,137 @@
|
|||||||
color: #ffffff;
|
color: #ffffff;
|
||||||
font-size: 1rem;
|
font-size: 1rem;
|
||||||
outline: none;
|
outline: none;
|
||||||
padding: 0.5rem 0;
|
padding: 0.5rem 1rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
.nexus-input::placeholder {
|
.nexus-input::placeholder {
|
||||||
color: rgba(255, 255, 255, 0.4);
|
color: rgba(255, 255, 255, 0.35);
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-nexus {
|
||||||
|
padding: 0.75rem 1.25rem;
|
||||||
|
border-radius: 10px;
|
||||||
|
font-size: 0.95rem;
|
||||||
|
font-weight: 600;
|
||||||
|
cursor: pointer;
|
||||||
|
transition: all 0.2s;
|
||||||
|
border: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-nexus.primary {
|
||||||
|
background: linear-gradient(135deg, #06b6d4 0%, #0891b2 100%);
|
||||||
|
color: #ffffff;
|
||||||
|
box-shadow: 0 4px 10px rgba(6, 182, 212, 0.25);
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-nexus:hover:not(:disabled) {
|
||||||
|
transform: translateY(-1px);
|
||||||
|
filter: brightness(1.1);
|
||||||
|
box-shadow: 0 4px 15px rgba(6, 182, 212, 0.4);
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-nexus:disabled {
|
||||||
|
background: rgba(255, 255, 255, 0.05);
|
||||||
|
color: rgba(255, 255, 255, 0.25);
|
||||||
|
box-shadow: none;
|
||||||
|
cursor: not-allowed;
|
||||||
}
|
}
|
||||||
|
|
||||||
.search-btn {
|
.search-btn {
|
||||||
border-radius: 25px !important;
|
width: 46px;
|
||||||
padding: 0.5rem 1.5rem !important;
|
height: 46px;
|
||||||
font-size: 0.95rem !important;
|
padding: 0 !important;
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
|
border-radius: 10px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.scope-selector {
|
.welcome-state {
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
gap: 0.75rem;
|
|
||||||
color: rgba(255, 255, 255, 0.7);
|
|
||||||
}
|
|
||||||
|
|
||||||
.nexus-select {
|
|
||||||
background: rgba(15, 23, 42, 0.6);
|
|
||||||
border: 1px solid rgba(255, 255, 255, 0.1);
|
|
||||||
color: #ffffff;
|
|
||||||
padding: 0.5rem 1.5rem 0.5rem 1rem;
|
|
||||||
border-radius: 20px;
|
|
||||||
outline: none;
|
|
||||||
cursor: pointer;
|
|
||||||
transition: all 0.3s ease;
|
|
||||||
}
|
|
||||||
|
|
||||||
.nexus-select:focus {
|
|
||||||
border-color: var(--nexus-primary, #6366f1);
|
|
||||||
}
|
|
||||||
|
|
||||||
.results-area {
|
|
||||||
min-height: 250px;
|
|
||||||
display: flex;
|
|
||||||
flex-direction: column;
|
|
||||||
justify-content: center;
|
|
||||||
}
|
|
||||||
|
|
||||||
.loading-state {
|
|
||||||
display: flex;
|
|
||||||
flex-direction: column;
|
|
||||||
align-items: center;
|
|
||||||
gap: 1.5rem;
|
|
||||||
color: rgba(255, 255, 255, 0.7);
|
|
||||||
}
|
|
||||||
|
|
||||||
.nexus-spinner {
|
|
||||||
width: 50px;
|
|
||||||
height: 50px;
|
|
||||||
border: 3px solid rgba(99, 102, 241, 0.1);
|
|
||||||
border-radius: 50%;
|
|
||||||
border-top-color: var(--nexus-primary, #6366f1);
|
|
||||||
animation: spin 1s linear infinite;
|
|
||||||
}
|
|
||||||
|
|
||||||
.welcome-state, .empty-state {
|
|
||||||
text-align: center;
|
text-align: center;
|
||||||
color: rgba(255, 255, 255, 0.5);
|
color: rgba(255, 255, 255, 0.5);
|
||||||
padding: 3rem 1rem;
|
padding: 4rem 2rem;
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
height: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
.welcome-icon {
|
.welcome-icon {
|
||||||
color: rgba(255, 255, 255, 0.25);
|
color: rgba(6, 182, 212, 0.4);
|
||||||
margin-bottom: 1.5rem;
|
margin-bottom: 1.5rem;
|
||||||
animation: pulse 2s infinite alternate;
|
filter: drop-shadow(0 0 10px rgba(6, 182, 212, 0.2));
|
||||||
|
animation: pulse 2.5s infinite alternate;
|
||||||
}
|
}
|
||||||
|
|
||||||
.welcome-state h3 {
|
.welcome-state h3 {
|
||||||
color: #ffffff;
|
color: #ffffff;
|
||||||
font-family: var(--nexus-font-sans);
|
font-size: 1.5rem;
|
||||||
margin: 0 0 0.5rem 0;
|
margin: 0 0 0.75rem 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
.welcome-state p, .empty-state p {
|
.welcome-state p {
|
||||||
max-width: 500px;
|
max-width: 550px;
|
||||||
margin: 0;
|
margin: 0;
|
||||||
font-size: 0.95rem;
|
font-size: 0.95rem;
|
||||||
line-height: 1.6;
|
line-height: 1.6;
|
||||||
}
|
}
|
||||||
|
|
||||||
.response-container {
|
|
||||||
display: flex;
|
|
||||||
flex-direction: column;
|
|
||||||
gap: 2.5rem;
|
|
||||||
animation: slideUp 0.4s ease-out;
|
|
||||||
}
|
|
||||||
|
|
||||||
.response-section h4, .citations-section h4 {
|
|
||||||
font-size: 1.1rem;
|
|
||||||
text-transform: uppercase;
|
|
||||||
letter-spacing: 1px;
|
|
||||||
color: rgba(255, 255, 255, 0.5);
|
|
||||||
margin: 0 0 1rem 0;
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
gap: 0.5rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
.answer-text {
|
|
||||||
font-size: 1.15rem;
|
|
||||||
line-height: 1.7;
|
|
||||||
color: #ffffff;
|
|
||||||
background: rgba(255, 255, 255, 0.03);
|
|
||||||
padding: 1.5rem;
|
|
||||||
border-radius: 12px;
|
|
||||||
border: 1px solid rgba(255, 255, 255, 0.05);
|
|
||||||
}
|
|
||||||
|
|
||||||
.citations-grid {
|
|
||||||
display: grid;
|
|
||||||
grid-template-columns: 1fr;
|
|
||||||
gap: 1.25rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
.citation-card {
|
|
||||||
background: rgba(15, 23, 42, 0.4);
|
|
||||||
border: 1px solid rgba(255, 255, 255, 0.05);
|
|
||||||
border-radius: 12px;
|
|
||||||
padding: 1.25rem;
|
|
||||||
transition: all 0.3s ease;
|
|
||||||
}
|
|
||||||
|
|
||||||
.citation-card:hover {
|
|
||||||
border-color: rgba(99, 102, 241, 0.3);
|
|
||||||
transform: translateY(-2px);
|
|
||||||
}
|
|
||||||
|
|
||||||
.citation-header {
|
|
||||||
display: flex;
|
|
||||||
justify-content: space-between;
|
|
||||||
align-items: center;
|
|
||||||
margin-bottom: 0.75rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
.source-badge {
|
|
||||||
font-size: 0.8rem;
|
|
||||||
font-weight: 600;
|
|
||||||
color: var(--nexus-primary, #6366f1);
|
|
||||||
background: rgba(99, 102, 241, 0.1);
|
|
||||||
padding: 0.25rem 0.75rem;
|
|
||||||
border-radius: 20px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.id-badge {
|
|
||||||
font-size: 0.75rem;
|
|
||||||
color: rgba(255, 255, 255, 0.4);
|
|
||||||
font-family: monospace;
|
|
||||||
}
|
|
||||||
|
|
||||||
.citation-body {
|
|
||||||
font-size: 0.95rem;
|
|
||||||
line-height: 1.5;
|
|
||||||
color: rgba(255, 255, 255, 0.8);
|
|
||||||
font-style: italic;
|
|
||||||
}
|
|
||||||
|
|
||||||
.btn-spinner {
|
.btn-spinner {
|
||||||
margin: 0;
|
width: 20px;
|
||||||
|
height: 20px;
|
||||||
|
border: 2px solid rgba(255, 255, 255, 0.1);
|
||||||
|
border-radius: 50%;
|
||||||
|
border-top-color: #ffffff;
|
||||||
|
animation: spin 0.8s linear infinite;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Animations */
|
/* Keyframe Animations */
|
||||||
@@keyframes fadeIn {
|
@@keyframes bubble-fade-in {
|
||||||
from { opacity: 0; transform: translateY(15px); }
|
0% { opacity: 0; transform: translateY(10px) scale(0.98); }
|
||||||
to { opacity: 1; transform: translateY(0); }
|
100% { opacity: 1; transform: translateY(0) scale(1); }
|
||||||
}
|
}
|
||||||
|
|
||||||
@@keyframes slideUp {
|
@@keyframes typing-bounce {
|
||||||
from { opacity: 0; transform: translateY(10px); }
|
0%, 100% { transform: translateY(0); }
|
||||||
to { opacity: 1; transform: translateY(0); }
|
50% { transform: translateY(-4px); }
|
||||||
|
}
|
||||||
|
|
||||||
|
@@keyframes pulse {
|
||||||
|
0% { transform: scale(0.96); opacity: 0.8; }
|
||||||
|
100% { transform: scale(1.04); opacity: 1; }
|
||||||
}
|
}
|
||||||
|
|
||||||
@@keyframes spin {
|
@@keyframes spin {
|
||||||
0% { transform: rotate(0deg); }
|
0% { transform: rotate(0deg); }
|
||||||
100% { transform: rotate(360deg); }
|
100% { transform: rotate(360deg); }
|
||||||
}
|
}
|
||||||
|
|
||||||
@@keyframes pulse {
|
|
||||||
0% { transform: scale(0.95); opacity: 0.7; }
|
|
||||||
100% { transform: scale(1.05); opacity: 1; }
|
|
||||||
}
|
|
||||||
</style>
|
</style>
|
||||||
|
|
||||||
@code {
|
@code {
|
||||||
private string _question = string.Empty;
|
private string _question = string.Empty;
|
||||||
private string _selectedBookId = string.Empty;
|
private string _selectedBookId = string.Empty;
|
||||||
private bool _isLoading;
|
private bool _isLoading;
|
||||||
private bool _hasSearched;
|
|
||||||
private GroundedResponseDto? _response;
|
|
||||||
private List<LastReadBookDto>? _books;
|
private List<LastReadBookDto>? _books;
|
||||||
|
private List<ChatMessage> _chatMessages = new();
|
||||||
|
|
||||||
|
public class ChatMessage
|
||||||
|
{
|
||||||
|
public string Id { get; set; } = Guid.NewGuid().ToString();
|
||||||
|
public string Sender { get; set; } = string.Empty; // "User" or "AI"
|
||||||
|
public string Text { get; set; } = string.Empty;
|
||||||
|
public DateTime Timestamp { get; set; } = DateTime.UtcNow;
|
||||||
|
public List<ResponseSegment> Segments { get; set; } = new();
|
||||||
|
public List<CitationDto> Citations { get; set; } = new();
|
||||||
|
}
|
||||||
|
|
||||||
|
public class ResponseSegment
|
||||||
|
{
|
||||||
|
public string Text { get; set; } = string.Empty;
|
||||||
|
public bool IsCitation { get; set; }
|
||||||
|
public string CitationId { get; set; } = string.Empty;
|
||||||
|
}
|
||||||
|
|
||||||
protected override async Task OnInitializedAsync()
|
protected override async Task OnInitializedAsync()
|
||||||
{
|
{
|
||||||
@@ -409,9 +592,18 @@
|
|||||||
{
|
{
|
||||||
if (string.IsNullOrWhiteSpace(_question) || _isLoading) return;
|
if (string.IsNullOrWhiteSpace(_question) || _isLoading) return;
|
||||||
|
|
||||||
|
var userQuestion = _question;
|
||||||
|
_question = string.Empty; // Clear input field immediately
|
||||||
_isLoading = true;
|
_isLoading = true;
|
||||||
_hasSearched = true;
|
|
||||||
_response = null;
|
// Add user query message
|
||||||
|
_chatMessages.Add(new ChatMessage
|
||||||
|
{
|
||||||
|
Sender = "User",
|
||||||
|
Text = userQuestion,
|
||||||
|
Segments = new List<ResponseSegment> { new ResponseSegment { Text = userQuestion, IsCitation = false } }
|
||||||
|
});
|
||||||
|
|
||||||
StateHasChanged();
|
StateHasChanged();
|
||||||
|
|
||||||
try
|
try
|
||||||
@@ -422,27 +614,41 @@
|
|||||||
ebookId = parsedId;
|
ebookId = parsedId;
|
||||||
}
|
}
|
||||||
|
|
||||||
var result = await KnowledgeService.AskQuestionAsync(_question, "tenantId", ebookId);
|
var authState = await AuthStateProvider.GetAuthenticationStateAsync();
|
||||||
|
var tenantId = authState.User.FindFirst("TenantId")?.Value ?? "global";
|
||||||
|
|
||||||
|
var result = await KnowledgeService.AskQuestionAsync(userQuestion, tenantId, ebookId);
|
||||||
if (result.IsSuccess)
|
if (result.IsSuccess)
|
||||||
{
|
{
|
||||||
_response = result.Value;
|
var response = result.Value;
|
||||||
|
_chatMessages.Add(new ChatMessage
|
||||||
|
{
|
||||||
|
Sender = "AI",
|
||||||
|
Text = response.Answer,
|
||||||
|
Segments = ParseSegments(response.Answer),
|
||||||
|
Citations = response.Citations
|
||||||
|
});
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
_response = new GroundedResponseDto
|
var errMsg = $"Error: {result.Errors.FirstOrDefault()?.Message ?? "An error occurred."}";
|
||||||
|
_chatMessages.Add(new ChatMessage
|
||||||
{
|
{
|
||||||
Answer = $"Error: {result.Errors.FirstOrDefault()?.Message ?? "An error occurred."}",
|
Sender = "AI",
|
||||||
Citations = new List<CitationDto>()
|
Text = errMsg,
|
||||||
};
|
Segments = new List<ResponseSegment> { new ResponseSegment { Text = errMsg, IsCitation = false } }
|
||||||
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
_response = new GroundedResponseDto
|
var errMsg = $"Network/API Error: {ex.Message}";
|
||||||
|
_chatMessages.Add(new ChatMessage
|
||||||
{
|
{
|
||||||
Answer = $"Network/API Error: {ex.Message}",
|
Sender = "AI",
|
||||||
Citations = new List<CitationDto>()
|
Text = errMsg,
|
||||||
};
|
Segments = new List<ResponseSegment> { new ResponseSegment { Text = errMsg, IsCitation = false } }
|
||||||
|
});
|
||||||
}
|
}
|
||||||
finally
|
finally
|
||||||
{
|
{
|
||||||
@@ -450,4 +656,77 @@
|
|||||||
StateHasChanged();
|
StateHasChanged();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private List<ResponseSegment> ParseSegments(string text)
|
||||||
|
{
|
||||||
|
var segments = new List<ResponseSegment>();
|
||||||
|
if (string.IsNullOrEmpty(text)) return segments;
|
||||||
|
|
||||||
|
// Matches [Source ID: some-id] OR raw GUIDs in brackets [e225e58f-7539-cd51-e0ab-82741ec7e65c]
|
||||||
|
var regex = new System.Text.RegularExpressions.Regex(
|
||||||
|
@"\[Source ID:\s*([^\]]+)\]|\[([a-fA-F0-9]{8}-[a-fA-F0-9]{4}-[a-fA-F0-9]{4}-[a-fA-F0-9]{4}-[a-fA-F0-9]{12})\]",
|
||||||
|
System.Text.RegularExpressions.RegexOptions.IgnoreCase);
|
||||||
|
var matches = regex.Matches(text);
|
||||||
|
|
||||||
|
int lastIndex = 0;
|
||||||
|
foreach (System.Text.RegularExpressions.Match match in matches)
|
||||||
|
{
|
||||||
|
if (match.Index > lastIndex)
|
||||||
|
{
|
||||||
|
segments.Add(new ResponseSegment
|
||||||
|
{
|
||||||
|
Text = text.Substring(lastIndex, match.Index - lastIndex),
|
||||||
|
IsCitation = false
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
var citationId = match.Groups[1].Success
|
||||||
|
? match.Groups[1].Value.Trim()
|
||||||
|
: match.Groups[2].Value.Trim();
|
||||||
|
|
||||||
|
segments.Add(new ResponseSegment
|
||||||
|
{
|
||||||
|
IsCitation = true,
|
||||||
|
CitationId = citationId
|
||||||
|
});
|
||||||
|
|
||||||
|
lastIndex = match.Index + match.Length;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (lastIndex < text.Length)
|
||||||
|
{
|
||||||
|
segments.Add(new ResponseSegment
|
||||||
|
{
|
||||||
|
Text = text.Substring(lastIndex),
|
||||||
|
IsCitation = false
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return segments;
|
||||||
|
}
|
||||||
|
|
||||||
|
private MarkupString RenderMarkdown(string text)
|
||||||
|
{
|
||||||
|
if (string.IsNullOrEmpty(text)) return new MarkupString(string.Empty);
|
||||||
|
|
||||||
|
// 1. HTML Encode to prevent XSS
|
||||||
|
var html = System.Net.WebUtility.HtmlEncode(text);
|
||||||
|
|
||||||
|
// 2. Bold: **text** -> <strong>text</strong>
|
||||||
|
html = System.Text.RegularExpressions.Regex.Replace(html, @"\*\*(.*?)\*\*", "<strong>$1</strong>");
|
||||||
|
|
||||||
|
// 3. Italic: *text* -> <em>text</em>
|
||||||
|
html = System.Text.RegularExpressions.Regex.Replace(html, @"\*(.*?)\*", "<em>$1</em>");
|
||||||
|
|
||||||
|
// 4. Code blocks: ```language ... ``` -> <pre class="nexus-code-block"><code>...</code></pre>
|
||||||
|
html = System.Text.RegularExpressions.Regex.Replace(html, @"```(?:[a-zA-Z0-9+#]+)?\s*([\s\S]*?)\s*```", "<pre class=\"nexus-code-block\"><code>$1</code></pre>");
|
||||||
|
|
||||||
|
// 5. Inline Code: `code` -> <code class="nexus-inline-code">code</code>
|
||||||
|
html = System.Text.RegularExpressions.Regex.Replace(html, @"`(.*?)`", "<code class=\"nexus-inline-code\">$1</code>");
|
||||||
|
|
||||||
|
// 6. Newlines: \n -> <br />
|
||||||
|
html = html.Replace("\n", "<br />");
|
||||||
|
|
||||||
|
return new MarkupString(html);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,370 @@
|
|||||||
|
@page "/serilog-demo"
|
||||||
|
@inject ILogger<SerilogDemo> Logger
|
||||||
|
@inject IJSRuntime JSRuntime
|
||||||
|
|
||||||
|
<div class="serilog-demo-container">
|
||||||
|
<div class="header-card">
|
||||||
|
<div class="header-content">
|
||||||
|
<NexusIcon Name="cpu" Size="36" Class="header-icon" />
|
||||||
|
<div class="header-text">
|
||||||
|
<h1>Serilog Logging Infrastructure</h1>
|
||||||
|
<p class="subtitle">Production-grade diagnostic pipeline for unified native & web logs</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="status-badge">
|
||||||
|
<span class="status-dot green"></span>
|
||||||
|
<span class="status-text">Pipeline Active</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="demo-grid">
|
||||||
|
<!-- Native .NET Logging Panel -->
|
||||||
|
<div class="control-card">
|
||||||
|
<div class="card-header">
|
||||||
|
<NexusIcon Name="terminal" Size="20" Class="card-icon" />
|
||||||
|
<h2>Native .NET Logs (C#)</h2>
|
||||||
|
</div>
|
||||||
|
<p class="card-desc">Trigger structured C# logs using Dependency Injected ILogger.</p>
|
||||||
|
<div class="btn-group">
|
||||||
|
<button class="btn btn-info" @onclick="LogInfo">
|
||||||
|
<NexusIcon Name="info" Size="16" />
|
||||||
|
Log Info
|
||||||
|
</button>
|
||||||
|
<button class="btn btn-warning" @onclick="LogWarning">
|
||||||
|
<NexusIcon Name="alert-triangle" Size="16" />
|
||||||
|
Log Warning
|
||||||
|
</button>
|
||||||
|
<button class="btn btn-error" @onclick="LogError">
|
||||||
|
<NexusIcon Name="x-circle" Size="16" />
|
||||||
|
Log Error Exception
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Blazor / JS Interop Bridge Panel -->
|
||||||
|
<div class="control-card">
|
||||||
|
<div class="card-header">
|
||||||
|
<NexusIcon Name="globe" Size="20" Class="card-icon js-icon" />
|
||||||
|
<h2>Blazor / JS WebView Logs</h2>
|
||||||
|
</div>
|
||||||
|
<p class="card-desc">Trigger logs from JavaScript to verify the interop error capture bridge.</p>
|
||||||
|
<div class="btn-group">
|
||||||
|
<button class="btn btn-js-info" @onclick="TriggerJsLog">
|
||||||
|
<NexusIcon Name="message-square" Size="16" />
|
||||||
|
Trigger console.log()
|
||||||
|
</button>
|
||||||
|
<button class="btn btn-js-error" @onclick="TriggerJsException">
|
||||||
|
<NexusIcon Name="zap" Size="16" />
|
||||||
|
Trigger JS Exception
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Active Log Config Panel -->
|
||||||
|
<div class="config-card">
|
||||||
|
<div class="card-header">
|
||||||
|
<NexusIcon Name="settings" Size="20" Class="card-icon" />
|
||||||
|
<h2>Pipeline Diagnostics</h2>
|
||||||
|
</div>
|
||||||
|
<div class="config-grid">
|
||||||
|
<div class="config-item">
|
||||||
|
<span class="label">Rolling Daily File Sandbox Path</span>
|
||||||
|
<span class="value code-value">AppDataDirectory/logs/log-*.txt</span>
|
||||||
|
</div>
|
||||||
|
<div class="config-item">
|
||||||
|
<span class="label">Active Configuration Provider</span>
|
||||||
|
<span class="value">Serilog.Settings.Configuration (appsettings.json)</span>
|
||||||
|
</div>
|
||||||
|
<div class="config-item">
|
||||||
|
<span class="label">Native Apple Console Sink</span>
|
||||||
|
<span class="value">Serilog.Sinks.Debug (conditional compilation)</span>
|
||||||
|
</div>
|
||||||
|
<div class="config-item">
|
||||||
|
<span class="label">Native Android Logcat Sink</span>
|
||||||
|
<span class="value">AndroidLogcatSink (direct JNI bindings)</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
.serilog-demo-container {
|
||||||
|
padding: 2rem;
|
||||||
|
max-width: 1200px;
|
||||||
|
margin: 0 auto;
|
||||||
|
color: #e2e8f0;
|
||||||
|
font-family: 'Inter', system-ui, -apple-system, sans-serif;
|
||||||
|
}
|
||||||
|
|
||||||
|
.header-card {
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
align-items: center;
|
||||||
|
background: linear-gradient(135deg, rgba(30, 41, 59, 0.7) 0%, rgba(15, 23, 42, 0.8) 100%);
|
||||||
|
border: 1px solid rgba(255, 255, 255, 0.08);
|
||||||
|
border-radius: 16px;
|
||||||
|
padding: 2rem;
|
||||||
|
margin-bottom: 2rem;
|
||||||
|
backdrop-filter: blur(12px);
|
||||||
|
box-shadow: 0 8px 32px 0 rgba(0, 0, 0, 0.37);
|
||||||
|
}
|
||||||
|
|
||||||
|
.header-content {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 1.5rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.header-icon {
|
||||||
|
color: #6366f1;
|
||||||
|
filter: drop-shadow(0 0 8px rgba(99, 102, 241, 0.5));
|
||||||
|
}
|
||||||
|
|
||||||
|
.header-text h1 {
|
||||||
|
font-size: 1.8rem;
|
||||||
|
font-weight: 700;
|
||||||
|
margin: 0;
|
||||||
|
background: linear-gradient(to right, #ffffff, #94a3b8);
|
||||||
|
-webkit-background-clip: text;
|
||||||
|
-webkit-text-fill-color: transparent;
|
||||||
|
}
|
||||||
|
|
||||||
|
.subtitle {
|
||||||
|
margin: 0.25rem 0 0 0;
|
||||||
|
color: #94a3b8;
|
||||||
|
font-size: 0.95rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.status-badge {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 0.5rem;
|
||||||
|
background: rgba(16, 185, 129, 0.1);
|
||||||
|
border: 1px solid rgba(16, 185, 129, 0.2);
|
||||||
|
padding: 0.5rem 1rem;
|
||||||
|
border-radius: 9999px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.status-dot {
|
||||||
|
width: 8px;
|
||||||
|
height: 8px;
|
||||||
|
border-radius: 50%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.status-dot.green {
|
||||||
|
background-color: #10b981;
|
||||||
|
box-shadow: 0 0 8px #10b981;
|
||||||
|
}
|
||||||
|
|
||||||
|
.status-text {
|
||||||
|
font-size: 0.85rem;
|
||||||
|
font-weight: 600;
|
||||||
|
color: #10b981;
|
||||||
|
}
|
||||||
|
|
||||||
|
.demo-grid {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: 1fr 1fr;
|
||||||
|
gap: 2rem;
|
||||||
|
margin-bottom: 2rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
@@media (max-width: 768px) {
|
||||||
|
.demo-grid {
|
||||||
|
grid-template-columns: 1fr;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.control-card {
|
||||||
|
background: rgba(30, 41, 59, 0.45);
|
||||||
|
border: 1px solid rgba(255, 255, 255, 0.05);
|
||||||
|
border-radius: 16px;
|
||||||
|
padding: 2rem;
|
||||||
|
backdrop-filter: blur(8px);
|
||||||
|
}
|
||||||
|
|
||||||
|
.card-header {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 0.75rem;
|
||||||
|
margin-bottom: 1rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.card-icon {
|
||||||
|
color: #6366f1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.js-icon {
|
||||||
|
color: #eab308;
|
||||||
|
}
|
||||||
|
|
||||||
|
.card-header h2 {
|
||||||
|
font-size: 1.25rem;
|
||||||
|
font-weight: 600;
|
||||||
|
margin: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.card-desc {
|
||||||
|
color: #94a3b8;
|
||||||
|
font-size: 0.9rem;
|
||||||
|
margin-bottom: 1.5rem;
|
||||||
|
line-height: 1.5;
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-group {
|
||||||
|
display: flex;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
gap: 1rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 0.5rem;
|
||||||
|
padding: 0.75rem 1.25rem;
|
||||||
|
border-radius: 8px;
|
||||||
|
font-size: 0.85rem;
|
||||||
|
font-weight: 600;
|
||||||
|
cursor: pointer;
|
||||||
|
transition: all 0.2s ease;
|
||||||
|
border: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-info {
|
||||||
|
background-color: rgba(99, 102, 241, 0.1);
|
||||||
|
color: #818cf8;
|
||||||
|
border: 1px solid rgba(99, 102, 241, 0.2);
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-info:hover {
|
||||||
|
background-color: #6366f1;
|
||||||
|
color: white;
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-warning {
|
||||||
|
background-color: rgba(245, 158, 11, 0.1);
|
||||||
|
color: #fbbf24;
|
||||||
|
border: 1px solid rgba(245, 158, 11, 0.2);
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-warning:hover {
|
||||||
|
background-color: #f59e0b;
|
||||||
|
color: white;
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-error {
|
||||||
|
background-color: rgba(239, 68, 68, 0.1);
|
||||||
|
color: #f87171;
|
||||||
|
border: 1px solid rgba(239, 68, 68, 0.2);
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-error:hover {
|
||||||
|
background-color: #ef4444;
|
||||||
|
color: white;
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-js-info {
|
||||||
|
background-color: rgba(234, 179, 8, 0.1);
|
||||||
|
color: #fef08a;
|
||||||
|
border: 1px solid rgba(234, 179, 8, 0.2);
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-js-info:hover {
|
||||||
|
background-color: #eab308;
|
||||||
|
color: #0f172a;
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-js-error {
|
||||||
|
background-color: rgba(236, 72, 153, 0.1);
|
||||||
|
color: #fbcfe8;
|
||||||
|
border: 1px solid rgba(236, 72, 153, 0.2);
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-js-error:hover {
|
||||||
|
background-color: #ec4899;
|
||||||
|
color: white;
|
||||||
|
}
|
||||||
|
|
||||||
|
.config-card {
|
||||||
|
background: rgba(15, 23, 42, 0.5);
|
||||||
|
border: 1px solid rgba(255, 255, 255, 0.05);
|
||||||
|
border-radius: 16px;
|
||||||
|
padding: 2rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.config-grid {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: 1fr 1fr;
|
||||||
|
gap: 1.5rem;
|
||||||
|
margin-top: 1.5rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
@@media (max-width: 768px) {
|
||||||
|
.config-grid {
|
||||||
|
grid-template-columns: 1fr;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.config-item {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 0.5rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.config-item .label {
|
||||||
|
font-size: 0.8rem;
|
||||||
|
color: #64748b;
|
||||||
|
text-transform: uppercase;
|
||||||
|
letter-spacing: 0.05em;
|
||||||
|
font-weight: 600;
|
||||||
|
}
|
||||||
|
|
||||||
|
.config-item .value {
|
||||||
|
font-size: 0.95rem;
|
||||||
|
color: #cbd5e1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.code-value {
|
||||||
|
font-family: 'Fira Code', 'Courier New', Courier, monospace;
|
||||||
|
background: rgba(0, 0, 0, 0.2);
|
||||||
|
padding: 0.25rem 0.5rem;
|
||||||
|
border-radius: 4px;
|
||||||
|
border: 1px solid rgba(255, 255, 255, 0.05);
|
||||||
|
word-break: break-all;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
|
||||||
|
@code {
|
||||||
|
private void LogInfo()
|
||||||
|
{
|
||||||
|
Logger.LogInformation("Structured native log triggered by user from SerilogDemo. Button: LogInfo");
|
||||||
|
}
|
||||||
|
|
||||||
|
private void LogWarning()
|
||||||
|
{
|
||||||
|
Logger.LogWarning("Potential warning log triggered from Blazor razor component at {Time}", DateTime.UtcNow);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void LogError()
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
throw new InvalidOperationException("Simulated native C# operation exception triggered in Diagnostic dashboard.");
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
Logger.LogError(ex, "Captured exception successfully in native Serilog pipeline!");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task TriggerJsLog()
|
||||||
|
{
|
||||||
|
await JSRuntime.InvokeVoidAsync("console.log", "Intercepted JS console statement from Blazor WebView interop trigger!");
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task TriggerJsException()
|
||||||
|
{
|
||||||
|
await JSRuntime.InvokeVoidAsync("eval", "throw new Error('Simulated runtime JS Exception triggered from Blazor UI button click!');");
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -2,16 +2,63 @@
|
|||||||
@attribute [Authorize]
|
@attribute [Authorize]
|
||||||
|
|
||||||
<div class="settings-page">
|
<div class="settings-page">
|
||||||
<h1>Ustawienia</h1>
|
<h1>Settings</h1>
|
||||||
<p>Konfiguracja Twojego konta i preferencji czytania.</p>
|
<p>Configure your account and application preferences.</p>
|
||||||
|
|
||||||
|
<div class="settings-section">
|
||||||
|
<h2>Diagnostics & System Logs</h2>
|
||||||
|
<p>Inspect native logging infrastructure, trigger custom logs, and trace WebView errors.</p>
|
||||||
|
<a class="diag-btn" href="/serilog-demo">
|
||||||
|
<NexusIcon Name="cpu" Size="16" />
|
||||||
|
Open Serilog Diagnostics Dashboard
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<style>
|
<style>
|
||||||
.settings-page {
|
.settings-page {
|
||||||
padding: 2rem;
|
padding: 2rem;
|
||||||
|
color: #e2e8f0;
|
||||||
|
font-family: 'Inter', sans-serif;
|
||||||
}
|
}
|
||||||
h1 {
|
h1 {
|
||||||
margin-bottom: 1rem;
|
margin-bottom: 0.5rem;
|
||||||
color: #fff;
|
color: #fff;
|
||||||
}
|
}
|
||||||
|
h2 {
|
||||||
|
margin-top: 2rem;
|
||||||
|
font-size: 1.2rem;
|
||||||
|
color: #fff;
|
||||||
|
margin-bottom: 0.5rem;
|
||||||
|
}
|
||||||
|
.settings-section {
|
||||||
|
background: rgba(30, 41, 59, 0.45);
|
||||||
|
border: 1px solid rgba(255, 255, 255, 0.05);
|
||||||
|
padding: 1.5rem;
|
||||||
|
border-radius: 12px;
|
||||||
|
margin-top: 1.5rem;
|
||||||
|
}
|
||||||
|
.settings-section p {
|
||||||
|
color: #94a3b8;
|
||||||
|
font-size: 0.9rem;
|
||||||
|
margin-bottom: 1.25rem;
|
||||||
|
}
|
||||||
|
.diag-btn {
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 0.5rem;
|
||||||
|
background: rgba(99, 102, 241, 0.1);
|
||||||
|
color: #818cf8;
|
||||||
|
border: 1px solid rgba(99, 102, 241, 0.2);
|
||||||
|
padding: 0.75rem 1.25rem;
|
||||||
|
border-radius: 8px;
|
||||||
|
text-decoration: none;
|
||||||
|
font-size: 0.85rem;
|
||||||
|
font-weight: 600;
|
||||||
|
transition: all 0.2s ease;
|
||||||
|
}
|
||||||
|
.diag-btn:hover {
|
||||||
|
background: #6366f1;
|
||||||
|
color: white;
|
||||||
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -6,6 +6,7 @@ public interface IReaderNavigationService
|
|||||||
int CurrentChapterIndex { get; }
|
int CurrentChapterIndex { get; }
|
||||||
int TotalChapters { get; }
|
int TotalChapters { get; }
|
||||||
string ChapterTitle { get; }
|
string ChapterTitle { get; }
|
||||||
|
string? PendingScrollBlockId { get; set; }
|
||||||
|
|
||||||
event Func<Task>? OnNavigationChanged;
|
event Func<Task>? OnNavigationChanged;
|
||||||
|
|
||||||
|
|||||||
@@ -7,5 +7,6 @@ public interface ISyncService
|
|||||||
Task<Result> InitializeAsync();
|
Task<Result> InitializeAsync();
|
||||||
Task<Result> UpdateProgressAsync(string pageId, Guid ebookId, double progress, string? chapterTitle, int chapterIndex);
|
Task<Result> UpdateProgressAsync(string pageId, Guid ebookId, double progress, string? chapterTitle, int chapterIndex);
|
||||||
event Func<string, DateTime, Task> OnProgressReceived;
|
event Func<string, DateTime, Task> OnProgressReceived;
|
||||||
|
event Func<string, double, Task>? OnIngestionProgressReceived;
|
||||||
Task DisposeAsync();
|
Task DisposeAsync();
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -249,6 +249,25 @@ public class IdentityService : IIdentityService
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void ClearCache()
|
||||||
|
{
|
||||||
|
_cachedProfile = null;
|
||||||
|
if (OnStateInvalidated != null)
|
||||||
|
{
|
||||||
|
_ = Task.Run(async () =>
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
await OnStateInvalidated.Invoke();
|
||||||
|
}
|
||||||
|
catch
|
||||||
|
{
|
||||||
|
// Ignore exceptions from event handlers
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private class LoginResponse
|
private class LoginResponse
|
||||||
{
|
{
|
||||||
public string TokenType { get; set; } = string.Empty;
|
public string TokenType { get; set; } = string.Empty;
|
||||||
|
|||||||
@@ -17,6 +17,8 @@ public sealed partial class KnowledgeCoordinator : IDisposable
|
|||||||
private readonly IReaderInteractionService _interactionService;
|
private readonly IReaderInteractionService _interactionService;
|
||||||
private readonly ILogger<KnowledgeCoordinator> _logger;
|
private readonly ILogger<KnowledgeCoordinator> _logger;
|
||||||
|
|
||||||
|
public string CurrentFullPageContent { get; private set; } = string.Empty;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Raised when the knowledge graph has been updated with new data.
|
/// Raised when the knowledge graph has been updated with new data.
|
||||||
/// Subscribers must return a Task to enable proper async handling.
|
/// Subscribers must return a Task to enable proper async handling.
|
||||||
@@ -77,6 +79,7 @@ public sealed partial class KnowledgeCoordinator : IDisposable
|
|||||||
{
|
{
|
||||||
if (string.IsNullOrWhiteSpace(fullContent)) return;
|
if (string.IsNullOrWhiteSpace(fullContent)) return;
|
||||||
|
|
||||||
|
CurrentFullPageContent = fullContent;
|
||||||
LogGeneratingGraph(tenantId);
|
LogGeneratingGraph(tenantId);
|
||||||
|
|
||||||
await _graphService.Clear();
|
await _graphService.Clear();
|
||||||
@@ -94,11 +97,15 @@ public sealed partial class KnowledgeCoordinator : IDisposable
|
|||||||
if (OnGraphUpdated != null)
|
if (OnGraphUpdated != null)
|
||||||
await OnGraphUpdated.Invoke(packet.Graph);
|
await OnGraphUpdated.Invoke(packet.Graph);
|
||||||
await _platformService.VibrateSuccessAsync();
|
await _platformService.VibrateSuccessAsync();
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
await _graphService.SetLoading(false);
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
|
await _graphService.SetLoading(false);
|
||||||
LogGraphError(ex, tenantId);
|
LogGraphError(ex, tenantId);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -144,6 +151,7 @@ public sealed partial class KnowledgeCoordinator : IDisposable
|
|||||||
|
|
||||||
public async Task ClearAsync()
|
public async Task ClearAsync()
|
||||||
{
|
{
|
||||||
|
CurrentFullPageContent = string.Empty;
|
||||||
await _graphService.Clear();
|
await _graphService.Clear();
|
||||||
await _quizService.SetQuiz(null, null);
|
await _quizService.SetQuiz(null, null);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -15,6 +15,7 @@ public class ReaderNavigationService : IReaderNavigationService
|
|||||||
public int CurrentChapterIndex { get; private set; } = 0;
|
public int CurrentChapterIndex { get; private set; } = 0;
|
||||||
public int TotalChapters { get; private set; } = 1;
|
public int TotalChapters { get; private set; } = 1;
|
||||||
public string ChapterTitle { get; private set; } = "Loading...";
|
public string ChapterTitle { get; private set; } = "Loading...";
|
||||||
|
public string? PendingScrollBlockId { get; set; }
|
||||||
|
|
||||||
public event Func<Task>? OnNavigationChanged;
|
public event Func<Task>? OnNavigationChanged;
|
||||||
|
|
||||||
|
|||||||
@@ -16,6 +16,7 @@ public class SyncService : ISyncService, IAsyncDisposable
|
|||||||
private CancellationTokenSource? _debounceCts;
|
private CancellationTokenSource? _debounceCts;
|
||||||
|
|
||||||
public event Func<string, DateTime, Task>? OnProgressReceived;
|
public event Func<string, DateTime, Task>? OnProgressReceived;
|
||||||
|
public event Func<string, double, Task>? OnIngestionProgressReceived;
|
||||||
|
|
||||||
public SyncService(
|
public SyncService(
|
||||||
HttpClient httpClient,
|
HttpClient httpClient,
|
||||||
@@ -50,9 +51,20 @@ public class SyncService : ISyncService, IAsyncDisposable
|
|||||||
_hubConnection.On<string, DateTime>("ProgressUpdated", async (pageId, timestamp) =>
|
_hubConnection.On<string, DateTime>("ProgressUpdated", async (pageId, timestamp) =>
|
||||||
{
|
{
|
||||||
// Note: In the future we might want to receive ebookId and progress here too
|
// Note: In the future we might want to receive ebookId and progress here too
|
||||||
|
if (pageId == _lastSentPageId)
|
||||||
|
{
|
||||||
|
_logger.LogDebug("[Sync] Ignoring self progress update for page {PageId}.", pageId);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
_lastSentPageId = pageId; // Prevent echoing back duplicate progress updates
|
||||||
if (OnProgressReceived != null) await OnProgressReceived(pageId, timestamp);
|
if (OnProgressReceived != null) await OnProgressReceived(pageId, timestamp);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
_hubConnection.On<string, double>("IngestionProgress", async (message, progress) =>
|
||||||
|
{
|
||||||
|
if (OnIngestionProgressReceived != null) await OnIngestionProgressReceived(message, progress);
|
||||||
|
});
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
await _hubConnection.StartAsync();
|
await _hubConnection.StartAsync();
|
||||||
@@ -71,6 +83,8 @@ public class SyncService : ISyncService, IAsyncDisposable
|
|||||||
{
|
{
|
||||||
if (pageId == _lastSentPageId) return Result.Ok();
|
if (pageId == _lastSentPageId) return Result.Ok();
|
||||||
|
|
||||||
|
_lastSentPageId = pageId;
|
||||||
|
|
||||||
// Proper trailing-edge debounce
|
// Proper trailing-edge debounce
|
||||||
_debounceCts?.Cancel();
|
_debounceCts?.Cancel();
|
||||||
_debounceCts = new CancellationTokenSource();
|
_debounceCts = new CancellationTokenSource();
|
||||||
@@ -86,8 +100,7 @@ public class SyncService : ISyncService, IAsyncDisposable
|
|||||||
|
|
||||||
if (_hubConnection?.State == HubConnectionState.Connected)
|
if (_hubConnection?.State == HubConnectionState.Connected)
|
||||||
{
|
{
|
||||||
await _hubConnection.SendAsync("UpdateProgress", pageId, ebookId, progress, chapterTitle, token);
|
await _hubConnection.SendAsync("UpdateProgress", pageId, ebookId, progress, chapterTitle, chapterIndex);
|
||||||
_lastSentPageId = pageId;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
catch (TaskCanceledException) { /* Ignored, user kept scrolling */ }
|
catch (TaskCanceledException) { /* Ignored, user kept scrolling */ }
|
||||||
|
|||||||
@@ -3,6 +3,110 @@ import * as d3 from 'https://esm.sh/d3@7';
|
|||||||
const getDisplayLabel = d => d.label.length > 20 ? d.label.substring(0, 17) + "..." : d.label;
|
const getDisplayLabel = d => d.label.length > 20 ? d.label.substring(0, 17) + "..." : d.label;
|
||||||
const getPillWidth = d => getDisplayLabel(d).length * 8 + 30;
|
const getPillWidth = d => getDisplayLabel(d).length * 8 + 30;
|
||||||
|
|
||||||
|
const getNodeType = d => {
|
||||||
|
if (d) {
|
||||||
|
if (d.type) {
|
||||||
|
const t = d.type.toLowerCase();
|
||||||
|
if (t === 'definition') return 'definition';
|
||||||
|
if (t === 'table') return 'table';
|
||||||
|
if (t === 'rule') return 'rule';
|
||||||
|
if (t === 'section') return 'section';
|
||||||
|
}
|
||||||
|
if (d.group) {
|
||||||
|
const g = d.group.toLowerCase();
|
||||||
|
if (g === 'definition') return 'definition';
|
||||||
|
if (g === 'table') return 'table';
|
||||||
|
if (g === 'rule') return 'rule';
|
||||||
|
if (g === 'section') return 'section';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
};
|
||||||
|
|
||||||
|
const getNodeGroup = d => {
|
||||||
|
if (d && d.group) {
|
||||||
|
const g = d.group.toLowerCase();
|
||||||
|
if (g === 'bridge') return 'bridge';
|
||||||
|
if (g === 'current') return 'current';
|
||||||
|
if (g === 'concept') return 'concept';
|
||||||
|
}
|
||||||
|
return 'concept'; // fallback
|
||||||
|
};
|
||||||
|
|
||||||
|
const getCategoryStyle = d => {
|
||||||
|
const type = getNodeType(d);
|
||||||
|
const group = getNodeGroup(d);
|
||||||
|
|
||||||
|
// 1. Rule (red/coral)
|
||||||
|
if (type === 'rule') {
|
||||||
|
return {
|
||||||
|
color: '#ff4646',
|
||||||
|
fill: 'rgba(255, 70, 70, 0.1)',
|
||||||
|
opacity: 0.8,
|
||||||
|
glowKey: 'rule',
|
||||||
|
textColor: '#ff8b8b'
|
||||||
|
};
|
||||||
|
}
|
||||||
|
// 2. Definition (gold/amber)
|
||||||
|
if (type === 'definition') {
|
||||||
|
return {
|
||||||
|
color: '#ffb03a',
|
||||||
|
fill: 'rgba(255, 176, 58, 0.1)',
|
||||||
|
opacity: 0.8,
|
||||||
|
glowKey: 'definition',
|
||||||
|
textColor: '#ffd18c'
|
||||||
|
};
|
||||||
|
}
|
||||||
|
// 3. Table (purple/magenta)
|
||||||
|
if (type === 'table') {
|
||||||
|
return {
|
||||||
|
color: '#d946ef',
|
||||||
|
fill: 'rgba(217, 70, 239, 0.1)',
|
||||||
|
opacity: 0.8,
|
||||||
|
glowKey: 'table',
|
||||||
|
textColor: '#f5d0fe'
|
||||||
|
};
|
||||||
|
}
|
||||||
|
// 4. Section (blue/indigo)
|
||||||
|
if (type === 'section') {
|
||||||
|
return {
|
||||||
|
color: '#3b82f6',
|
||||||
|
fill: 'rgba(59, 130, 246, 0.1)',
|
||||||
|
opacity: 0.8,
|
||||||
|
glowKey: 'section',
|
||||||
|
textColor: '#93c5fd'
|
||||||
|
};
|
||||||
|
}
|
||||||
|
// 5. Bridge (cyan/comparison)
|
||||||
|
if (group === 'bridge') {
|
||||||
|
return {
|
||||||
|
color: '#06b6d4',
|
||||||
|
fill: 'rgba(6, 182, 212, 0.1)',
|
||||||
|
opacity: 0.7,
|
||||||
|
glowKey: 'bridge',
|
||||||
|
textColor: '#67e8f9'
|
||||||
|
};
|
||||||
|
}
|
||||||
|
// 6. Current (active/focus landmark - neon green)
|
||||||
|
if (group === 'current') {
|
||||||
|
return {
|
||||||
|
color: 'var(--nexus-neon)',
|
||||||
|
fill: 'rgba(0, 255, 153, 0.15)',
|
||||||
|
opacity: 0.9,
|
||||||
|
glowKey: 'current',
|
||||||
|
textColor: '#ffffff'
|
||||||
|
};
|
||||||
|
}
|
||||||
|
// 7. Concept / Default (subtle cool steel blue/teal)
|
||||||
|
return {
|
||||||
|
color: '#00d2c4',
|
||||||
|
fill: 'rgba(0, 210, 196, 0.05)',
|
||||||
|
opacity: 0.4,
|
||||||
|
glowKey: 'concept',
|
||||||
|
textColor: '#e0e0e0'
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
let simulation;
|
let simulation;
|
||||||
let zoomBehavior;
|
let zoomBehavior;
|
||||||
let svgElement;
|
let svgElement;
|
||||||
@@ -24,8 +128,10 @@ export function mount(containerId, data, dotNetHelper) {
|
|||||||
.attr("height", "100%")
|
.attr("height", "100%")
|
||||||
.style("background", "radial-gradient(circle, #1a1a1a 0%, #121212 100%)");
|
.style("background", "radial-gradient(circle, #1a1a1a 0%, #121212 100%)");
|
||||||
|
|
||||||
// Radial gradient for Nebula effect
|
// Radial gradients for Nebula effects
|
||||||
const defs = svgElement.append("defs");
|
const defs = svgElement.append("defs");
|
||||||
|
|
||||||
|
// Fallback radial gradient for legacy nebulaGlow
|
||||||
const radialGradient = defs.append("radialGradient")
|
const radialGradient = defs.append("radialGradient")
|
||||||
.attr("id", "nebulaGlow")
|
.attr("id", "nebulaGlow")
|
||||||
.attr("cx", "50%")
|
.attr("cx", "50%")
|
||||||
@@ -34,6 +140,26 @@ export function mount(containerId, data, dotNetHelper) {
|
|||||||
radialGradient.append("stop").attr("offset", "0%").attr("stop-color", "var(--nexus-neon)").attr("stop-opacity", 1);
|
radialGradient.append("stop").attr("offset", "0%").attr("stop-color", "var(--nexus-neon)").attr("stop-opacity", 1);
|
||||||
radialGradient.append("stop").attr("offset", "100%").attr("stop-color", "var(--nexus-neon)").attr("stop-opacity", 0);
|
radialGradient.append("stop").attr("offset", "100%").attr("stop-color", "var(--nexus-neon)").attr("stop-opacity", 0);
|
||||||
|
|
||||||
|
const colors = {
|
||||||
|
'rule': '#ff4646',
|
||||||
|
'definition': '#ffb03a',
|
||||||
|
'table': '#d946ef',
|
||||||
|
'section': '#3b82f6',
|
||||||
|
'bridge': '#06b6d4',
|
||||||
|
'current': 'var(--nexus-neon)',
|
||||||
|
'concept': '#00d2c4'
|
||||||
|
};
|
||||||
|
|
||||||
|
Object.entries(colors).forEach(([key, color]) => {
|
||||||
|
const radGrad = defs.append("radialGradient")
|
||||||
|
.attr("id", `nebulaGlow-${key}`)
|
||||||
|
.attr("cx", "50%")
|
||||||
|
.attr("cy", "50%")
|
||||||
|
.attr("r", "50%");
|
||||||
|
radGrad.append("stop").attr("offset", "0%").attr("stop-color", color).attr("stop-opacity", 1);
|
||||||
|
radGrad.append("stop").attr("offset", "100%").attr("stop-color", color).attr("stop-opacity", 0);
|
||||||
|
});
|
||||||
|
|
||||||
// Root Group for Zoom
|
// Root Group for Zoom
|
||||||
rootGroup = svgElement.append("g").attr("class", "zoom-containment");
|
rootGroup = svgElement.append("g").attr("class", "zoom-containment");
|
||||||
|
|
||||||
@@ -135,21 +261,33 @@ export function updateData(data) {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// Sanitize links to filter out any references to non-existent nodes
|
||||||
|
const nodeIds = new Set(data.nodes.map(n => n.id));
|
||||||
|
const validLinks = (data.links || []).filter(l => {
|
||||||
|
const srcId = typeof l.source === 'object' ? l.source.id : l.source;
|
||||||
|
const tgtId = typeof l.target === 'object' ? l.target.id : l.target;
|
||||||
|
return nodeIds.has(srcId) && nodeIds.has(tgtId);
|
||||||
|
});
|
||||||
|
|
||||||
// Update Links
|
// Update Links
|
||||||
link = rootGroup.select(".links-layer")
|
link = rootGroup.select(".links-layer")
|
||||||
.selectAll("path")
|
.selectAll("path")
|
||||||
.data(data.links, d => d.source + "-" + d.target + "-" + d.relationType)
|
.data(validLinks, d => {
|
||||||
|
const srcId = typeof d.source === 'object' ? d.source.id : d.source;
|
||||||
|
const tgtId = typeof d.target === 'object' ? d.target.id : d.target;
|
||||||
|
return srcId + "-" + tgtId + "-" + d.type;
|
||||||
|
})
|
||||||
.join(
|
.join(
|
||||||
enter => enter.append("path")
|
enter => enter.append("path")
|
||||||
.attr("stroke", d => {
|
.attr("stroke", d => {
|
||||||
if (d.relationType === 'Defines') return 'var(--nexus-accent)';
|
if (d.type === 'Defines' || d.type === 'maps_to') return 'var(--nexus-accent, #00ffaa)';
|
||||||
if (d.relationType === 'Next') return 'rgba(255,255,255,0.2)';
|
if (d.type === 'Next' || d.type === 'relates_to') return 'rgba(255,255,255,0.2)';
|
||||||
if (d.relationType === 'Contains') return 'var(--nexus-neon)';
|
if (d.type === 'Contains' || d.type === 'contains') return 'var(--nexus-neon)';
|
||||||
return 'rgba(255,255,255,0.1)';
|
return 'rgba(255,255,255,0.1)';
|
||||||
})
|
})
|
||||||
.attr("fill", "none")
|
.attr("fill", "none")
|
||||||
.attr("stroke-width", d => d.relationType === 'Defines' ? 2 : 1)
|
.attr("stroke-width", d => (d.type === 'Defines' || d.type === 'maps_to') ? 2 : 1)
|
||||||
.attr("stroke-dasharray", d => d.relationType === 'References' ? "5,5" : "0")
|
.attr("stroke-dasharray", d => d.type === 'References' ? "5,5" : "0")
|
||||||
.style("opacity", 0)
|
.style("opacity", 0)
|
||||||
.call(enter => enter.transition().duration(500).style("opacity", 1)),
|
.call(enter => enter.transition().duration(500).style("opacity", 1)),
|
||||||
update => update,
|
update => update,
|
||||||
@@ -174,13 +312,8 @@ export function updateData(data) {
|
|||||||
|
|
||||||
g.append("circle")
|
g.append("circle")
|
||||||
.attr("r", 30)
|
.attr("r", 30)
|
||||||
.attr("fill", d => {
|
.attr("fill", d => `url(#nebulaGlow-${getCategoryStyle(d).glowKey})`)
|
||||||
if (d.type === 'Definition') return 'var(--nexus-accent)';
|
.attr("opacity", d => getCategoryStyle(d).opacity);
|
||||||
if (d.type === 'Table') return 'var(--nexus-neon)';
|
|
||||||
if (d.type === 'Rule') return '#ff4444';
|
|
||||||
return "url(#nebulaGlow)";
|
|
||||||
})
|
|
||||||
.attr("opacity", d => d.group === 'current' ? 0.6 : 0.2);
|
|
||||||
|
|
||||||
g.append("rect")
|
g.append("rect")
|
||||||
.attr("class", "node-pill")
|
.attr("class", "node-pill")
|
||||||
@@ -189,23 +322,20 @@ export function updateData(data) {
|
|||||||
.attr("width", d => getPillWidth(d))
|
.attr("width", d => getPillWidth(d))
|
||||||
.attr("height", 30)
|
.attr("height", 30)
|
||||||
.attr("rx", 15)
|
.attr("rx", 15)
|
||||||
.attr("fill", "rgba(20, 20, 20, 0.9)")
|
.attr("fill", "rgba(20, 20, 20, 0.95)")
|
||||||
.attr("stroke", d => {
|
.attr("stroke", d => getCategoryStyle(d).color)
|
||||||
if (d.type === 'Definition') return 'var(--nexus-accent)';
|
.attr("stroke-width", d => getNodeGroup(d) === 'current' ? 2 : 1.2);
|
||||||
if (d.type === 'Rule') return '#ff4444';
|
|
||||||
return "rgba(255, 255, 255, 0.1)";
|
|
||||||
})
|
|
||||||
.attr("stroke-width", 1);
|
|
||||||
|
|
||||||
g.append("text")
|
g.append("text")
|
||||||
.text(d => getDisplayLabel(d))
|
.text(d => getDisplayLabel(d))
|
||||||
.attr("text-anchor", "middle")
|
.attr("text-anchor", "middle")
|
||||||
.attr("y", 5)
|
.attr("y", 5)
|
||||||
.attr("fill", d => d.type === 'Definition' ? 'var(--nexus-accent)' : '#ccc')
|
.attr("fill", d => getCategoryStyle(d).textColor)
|
||||||
.attr("font-size", "0.8rem");
|
.attr("font-size", "0.8rem")
|
||||||
|
.attr("font-weight", d => getNodeGroup(d) === 'current' ? '600' : 'normal');
|
||||||
|
|
||||||
g.append("title")
|
g.append("title")
|
||||||
.text(d => d.label);
|
.text(d => d.description ? `${d.label}\n\n${d.description}` : d.label);
|
||||||
|
|
||||||
g.transition().duration(500).style("opacity", 1);
|
g.transition().duration(500).style("opacity", 1);
|
||||||
|
|
||||||
@@ -216,7 +346,7 @@ export function updateData(data) {
|
|||||||
);
|
);
|
||||||
|
|
||||||
simulation.nodes(data.nodes);
|
simulation.nodes(data.nodes);
|
||||||
simulation.force("link").links(data.links);
|
simulation.force("link").links(validLinks);
|
||||||
simulation.alpha(0.5).restart();
|
simulation.alpha(0.5).restart();
|
||||||
|
|
||||||
// Trigger zoom to fit after a short delay to allow simulation to settle
|
// Trigger zoom to fit after a short delay to allow simulation to settle
|
||||||
@@ -398,6 +528,15 @@ export function clear() {
|
|||||||
}
|
}
|
||||||
simulation.nodes([]);
|
simulation.nodes([]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Reset selections
|
||||||
|
link = null;
|
||||||
|
node = null;
|
||||||
|
|
||||||
|
// Reset D3 zoom transform to clean identity state
|
||||||
|
if (svgElement && zoomBehavior) {
|
||||||
|
svgElement.call(zoomBehavior.transform, d3.zoomIdentity);
|
||||||
|
}
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
console.warn("Failed to clear force simulation safely:", e);
|
console.warn("Failed to clear force simulation safely:", e);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -50,7 +50,9 @@ builder.Services.AddSingleton<IDbContextFactory<AppDbContext>>(new ThrowingDbCon
|
|||||||
builder.Services.AddSingleton<IEmbeddingGenerator<string, Embedding<float>>>(new ThrowingEmbeddingGenerator());
|
builder.Services.AddSingleton<IEmbeddingGenerator<string, Embedding<float>>>(new ThrowingEmbeddingGenerator());
|
||||||
builder.Services.AddSingleton<IBookStorageService>(new ThrowingBookStorageService());
|
builder.Services.AddSingleton<IBookStorageService>(new ThrowingBookStorageService());
|
||||||
builder.Services.AddSingleton<IEbookRepository>(new ThrowingEbookRepository());
|
builder.Services.AddSingleton<IEbookRepository>(new ThrowingEbookRepository());
|
||||||
|
builder.Services.AddSingleton<IQuizResultRepository>(new ThrowingQuizResultRepository());
|
||||||
builder.Services.AddSingleton<ISyncBroadcaster>(new ThrowingSyncBroadcaster());
|
builder.Services.AddSingleton<ISyncBroadcaster>(new ThrowingSyncBroadcaster());
|
||||||
|
builder.Services.AddSingleton<IEpubExtractor>(new ThrowingEpubExtractor());
|
||||||
|
|
||||||
builder.Services.AddApplication();
|
builder.Services.AddApplication();
|
||||||
builder.Services.AddScoped<IEpubReader, WasmEpubReader>();
|
builder.Services.AddScoped<IEpubReader, WasmEpubReader>();
|
||||||
@@ -88,6 +90,16 @@ public class ThrowingEbookRepository : IEbookRepository
|
|||||||
public Task<Author?> FindAuthorByNameAsync(string name, CancellationToken cancellationToken = default) => throw new NotSupportedException(ErrorMessage);
|
public Task<Author?> FindAuthorByNameAsync(string name, CancellationToken cancellationToken = default) => throw new NotSupportedException(ErrorMessage);
|
||||||
public void AddAuthor(Author author) => throw new NotSupportedException(ErrorMessage);
|
public void AddAuthor(Author author) => throw new NotSupportedException(ErrorMessage);
|
||||||
public void AddEbook(Ebook ebook) => throw new NotSupportedException(ErrorMessage);
|
public void AddEbook(Ebook ebook) => throw new NotSupportedException(ErrorMessage);
|
||||||
|
public Task<Ebook?> FindByIdAsync(Guid id, CancellationToken cancellationToken = default) => throw new NotSupportedException(ErrorMessage);
|
||||||
|
public Task<int> SaveChangesAsync(CancellationToken cancellationToken = default) => throw new NotSupportedException(ErrorMessage);
|
||||||
|
}
|
||||||
|
|
||||||
|
public class ThrowingQuizResultRepository : IQuizResultRepository
|
||||||
|
{
|
||||||
|
private const string ErrorMessage = "QuizResult repository operations are not supported in the WASM client. Use the API endpoint for data access.";
|
||||||
|
|
||||||
|
public Task<NexusUser?> FindUserByIdAsync(string userId, CancellationToken cancellationToken = default) => throw new NotSupportedException(ErrorMessage);
|
||||||
|
public void AddQuizResult(QuizResult quizResult) => throw new NotSupportedException(ErrorMessage);
|
||||||
public Task<int> SaveChangesAsync(CancellationToken cancellationToken = default) => throw new NotSupportedException(ErrorMessage);
|
public Task<int> SaveChangesAsync(CancellationToken cancellationToken = default) => throw new NotSupportedException(ErrorMessage);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -99,3 +111,9 @@ public class ThrowingSyncBroadcaster : ISyncBroadcaster
|
|||||||
public Task BroadcastIngestionProgressAsync(string userId, string message, double progress, CancellationToken cancellationToken = default)
|
public Task BroadcastIngestionProgressAsync(string userId, string message, double progress, CancellationToken cancellationToken = default)
|
||||||
=> throw new NotSupportedException("Real-time broadcasting can only be performed by the server.");
|
=> throw new NotSupportedException("Real-time broadcasting can only be performed by the server.");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public class ThrowingEpubExtractor : IEpubExtractor
|
||||||
|
{
|
||||||
|
public Task<FluentResults.Result<List<string>>> ExtractChaptersTextAsync(string relativePath, CancellationToken cancellationToken = default)
|
||||||
|
=> throw new NotSupportedException("EPUB text extraction is not supported in the WASM client.");
|
||||||
|
}
|
||||||
|
|||||||
@@ -106,7 +106,8 @@ builder.Services.AddAuthentication(options =>
|
|||||||
|
|
||||||
builder.Services.AddIdentityApiEndpoints<NexusUser>()
|
builder.Services.AddIdentityApiEndpoints<NexusUser>()
|
||||||
.AddRoles<IdentityRole>()
|
.AddRoles<IdentityRole>()
|
||||||
.AddEntityFrameworkStores<AppDbContext>();
|
.AddEntityFrameworkStores<AppDbContext>()
|
||||||
|
.AddClaimsPrincipalFactory<NexusReader.Web.Services.CustomUserClaimsPrincipalFactory>();
|
||||||
|
|
||||||
builder.Services.ConfigureApplicationCookie(options =>
|
builder.Services.ConfigureApplicationCookie(options =>
|
||||||
{
|
{
|
||||||
@@ -194,6 +195,7 @@ using (var scope = app.Services.CreateScope())
|
|||||||
|
|
||||||
await dbContext.Database.MigrateAsync();
|
await dbContext.Database.MigrateAsync();
|
||||||
await DbInitializer.SeedAsync(services);
|
await DbInitializer.SeedAsync(services);
|
||||||
|
await TriggerBackgroundProcessingForUnindexedBooksAsync(services);
|
||||||
|
|
||||||
if (logger.IsEnabled(LogLevel.Information))
|
if (logger.IsEnabled(LogLevel.Information))
|
||||||
{
|
{
|
||||||
@@ -337,13 +339,16 @@ app.MapPost("/api/library/ingest", async ([FromBody] IngestEbookRequest request,
|
|||||||
? Convert.FromBase64String(request.CoverImageBase64)
|
? Convert.FromBase64String(request.CoverImageBase64)
|
||||||
: null;
|
: null;
|
||||||
|
|
||||||
|
var tenantId = user.FindFirst("TenantId")?.Value ?? "global";
|
||||||
|
|
||||||
var command = new IngestEbookCommand(
|
var command = new IngestEbookCommand(
|
||||||
request.Title,
|
request.Title,
|
||||||
request.AuthorName,
|
request.AuthorName,
|
||||||
coverData,
|
coverData,
|
||||||
epubData,
|
epubData,
|
||||||
request.Description,
|
request.Description,
|
||||||
userId
|
userId,
|
||||||
|
tenantId
|
||||||
);
|
);
|
||||||
|
|
||||||
var result = await mediator.Send(command);
|
var result = await mediator.Send(command);
|
||||||
@@ -563,6 +568,50 @@ app.MapRazorComponents<App>()
|
|||||||
|
|
||||||
app.Run();
|
app.Run();
|
||||||
|
|
||||||
|
async Task TriggerBackgroundProcessingForUnindexedBooksAsync(IServiceProvider services)
|
||||||
|
{
|
||||||
|
var logger = services.GetRequiredService<ILogger<Program>>();
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var dbContextFactory = services.GetRequiredService<IDbContextFactory<NexusReader.Data.Persistence.AppDbContext>>();
|
||||||
|
using var dbContext = await dbContextFactory.CreateDbContextAsync();
|
||||||
|
|
||||||
|
var unindexedEbooks = await dbContext.Ebooks
|
||||||
|
.Where(e => !e.IsReadyForReading)
|
||||||
|
.ToListAsync();
|
||||||
|
|
||||||
|
if (unindexedEbooks.Any())
|
||||||
|
{
|
||||||
|
logger.LogInformation("[Startup] Found {Count} un-indexed ebooks. Triggering background processing...", unindexedEbooks.Count);
|
||||||
|
|
||||||
|
foreach (var ebook in unindexedEbooks)
|
||||||
|
{
|
||||||
|
logger.LogInformation("[Startup] Queuing background processing for ebook: '{Title}' ({Id})", ebook.Title, ebook.Id);
|
||||||
|
|
||||||
|
_ = Task.Run(async () =>
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
using var scope = services.CreateScope();
|
||||||
|
var scopedMediator = scope.ServiceProvider.GetRequiredService<IMediator>();
|
||||||
|
await scopedMediator.Send(new ProcessEbookCommand(ebook.Id, ebook.UserId, ebook.TenantId));
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
using var scope = services.CreateScope();
|
||||||
|
var scopedLogger = scope.ServiceProvider.GetRequiredService<ILogger<Program>>();
|
||||||
|
scopedLogger.LogError(ex, "Failed to run background processing for ebook {EbookId} on startup", ebook.Id);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
logger.LogError(ex, "Error checking or triggering background processing for unindexed books on startup.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public record KnowledgeRequest(string Text, Guid? EbookId = null);
|
public record KnowledgeRequest(string Text, Guid? EbookId = null);
|
||||||
public record GroundednessRequest(string Answer, string Context);
|
public record GroundednessRequest(string Answer, string Context);
|
||||||
public record SemanticSearchRequest(string QueryText, int Limit = 5);
|
public record SemanticSearchRequest(string QueryText, int Limit = 5);
|
||||||
|
|||||||
@@ -0,0 +1,28 @@
|
|||||||
|
using System.Security.Claims;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using Microsoft.AspNetCore.Identity;
|
||||||
|
using Microsoft.Extensions.Options;
|
||||||
|
using NexusReader.Domain.Entities;
|
||||||
|
|
||||||
|
namespace NexusReader.Web.Services;
|
||||||
|
|
||||||
|
public class CustomUserClaimsPrincipalFactory : UserClaimsPrincipalFactory<NexusUser, IdentityRole>
|
||||||
|
{
|
||||||
|
public CustomUserClaimsPrincipalFactory(
|
||||||
|
UserManager<NexusUser> userManager,
|
||||||
|
RoleManager<IdentityRole> roleManager,
|
||||||
|
IOptions<IdentityOptions> optionsAccessor)
|
||||||
|
: base(userManager, roleManager, optionsAccessor)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override async Task<ClaimsIdentity> GenerateClaimsAsync(NexusUser user)
|
||||||
|
{
|
||||||
|
var identity = await base.GenerateClaimsAsync(user);
|
||||||
|
if (!string.IsNullOrEmpty(user.TenantId))
|
||||||
|
{
|
||||||
|
identity.AddClaim(new Claim("TenantId", user.TenantId));
|
||||||
|
}
|
||||||
|
return identity;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -118,4 +118,22 @@ public class ServerIdentityService : IIdentityService
|
|||||||
|
|
||||||
return Result.Ok(result.Value);
|
return Result.Ok(result.Value);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void ClearCache()
|
||||||
|
{
|
||||||
|
if (OnStateInvalidated != null)
|
||||||
|
{
|
||||||
|
_ = Task.Run(async () =>
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
await OnStateInvalidated.Invoke();
|
||||||
|
}
|
||||||
|
catch
|
||||||
|
{
|
||||||
|
// Ignore
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,99 @@
|
|||||||
|
using System;
|
||||||
|
using System.Threading;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using FluentAssertions;
|
||||||
|
using Microsoft.Data.Sqlite;
|
||||||
|
using Microsoft.EntityFrameworkCore;
|
||||||
|
using Moq;
|
||||||
|
using NexusReader.Application.Abstractions.Persistence;
|
||||||
|
using NexusReader.Application.Commands.Quiz;
|
||||||
|
using NexusReader.Data.Persistence;
|
||||||
|
using NexusReader.Domain.Entities;
|
||||||
|
using NexusReader.Infrastructure.Persistence;
|
||||||
|
using Xunit;
|
||||||
|
|
||||||
|
namespace NexusReader.Application.Tests.Commands;
|
||||||
|
|
||||||
|
public class SubmitQuizResultCommandHandlerTests
|
||||||
|
{
|
||||||
|
private readonly Mock<IQuizResultRepository> _repositoryMock;
|
||||||
|
|
||||||
|
public SubmitQuizResultCommandHandlerTests()
|
||||||
|
{
|
||||||
|
_repositoryMock = new Mock<IQuizResultRepository>();
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public async Task Handle_WithValidRequest_PersistsQuizResultToDatabase()
|
||||||
|
{
|
||||||
|
// Arrange
|
||||||
|
var user = new NexusUser
|
||||||
|
{
|
||||||
|
Id = "user-abc",
|
||||||
|
UserName = "testuser",
|
||||||
|
Email = "test@example.com",
|
||||||
|
TenantId = "tenant-xyz",
|
||||||
|
SubscriptionPlanId = 1
|
||||||
|
};
|
||||||
|
|
||||||
|
_repositoryMock.Setup(r => r.FindUserByIdAsync("user-abc", It.IsAny<CancellationToken>()))
|
||||||
|
.ReturnsAsync(user);
|
||||||
|
|
||||||
|
QuizResult? capturedQuizResult = null;
|
||||||
|
_repositoryMock.Setup(r => r.AddQuizResult(It.IsAny<QuizResult>()))
|
||||||
|
.Callback<QuizResult>(q => capturedQuizResult = q);
|
||||||
|
|
||||||
|
_repositoryMock.Setup(r => r.SaveChangesAsync(It.IsAny<CancellationToken>()))
|
||||||
|
.ReturnsAsync(1);
|
||||||
|
|
||||||
|
var command = new SubmitQuizResultCommand(
|
||||||
|
UserId: "user-abc",
|
||||||
|
Topic: "Sprawdzian: .NET 10",
|
||||||
|
Score: 4,
|
||||||
|
TotalQuestions: 5
|
||||||
|
);
|
||||||
|
|
||||||
|
var handler = new SubmitQuizResultCommandHandler(_repositoryMock.Object);
|
||||||
|
|
||||||
|
// Act
|
||||||
|
var result = await handler.Handle(command, CancellationToken.None);
|
||||||
|
|
||||||
|
// Assert
|
||||||
|
result.IsSuccess.Should().BeTrue();
|
||||||
|
capturedQuizResult.Should().NotBeNull();
|
||||||
|
capturedQuizResult!.Topic.Should().Be("Sprawdzian: .NET 10");
|
||||||
|
capturedQuizResult.Score.Should().Be(4);
|
||||||
|
capturedQuizResult.TotalQuestions.Should().Be(5);
|
||||||
|
capturedQuizResult.TenantId.Should().Be("tenant-xyz");
|
||||||
|
|
||||||
|
_repositoryMock.Verify(r => r.AddQuizResult(It.IsAny<QuizResult>()), Times.Once);
|
||||||
|
_repositoryMock.Verify(r => r.SaveChangesAsync(It.IsAny<CancellationToken>()), Times.Once);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public async Task Handle_WithNonExistentUser_ReturnsFailureResult()
|
||||||
|
{
|
||||||
|
// Arrange
|
||||||
|
_repositoryMock.Setup(r => r.FindUserByIdAsync("non-existent", It.IsAny<CancellationToken>()))
|
||||||
|
.ReturnsAsync((NexusUser?)null);
|
||||||
|
|
||||||
|
var command = new SubmitQuizResultCommand(
|
||||||
|
UserId: "non-existent",
|
||||||
|
Topic: "Sprawdzian: .NET 10",
|
||||||
|
Score: 4,
|
||||||
|
TotalQuestions: 5
|
||||||
|
);
|
||||||
|
|
||||||
|
var handler = new SubmitQuizResultCommandHandler(_repositoryMock.Object);
|
||||||
|
|
||||||
|
// Act
|
||||||
|
var result = await handler.Handle(command, CancellationToken.None);
|
||||||
|
|
||||||
|
// Assert
|
||||||
|
result.IsFailed.Should().BeTrue();
|
||||||
|
result.Errors.Should().ContainSingle(e => e.Message == "User not found.");
|
||||||
|
|
||||||
|
_repositoryMock.Verify(r => r.AddQuizResult(It.IsAny<QuizResult>()), Times.Never);
|
||||||
|
_repositoryMock.Verify(r => r.SaveChangesAsync(It.IsAny<CancellationToken>()), Times.Never);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,58 @@
|
|||||||
|
using System;
|
||||||
|
using System.IO;
|
||||||
|
using System.Text.Json;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using Microsoft.EntityFrameworkCore;
|
||||||
|
using NexusReader.Data.Persistence;
|
||||||
|
using Xunit;
|
||||||
|
|
||||||
|
namespace NexusReader.Application.Tests.Queries;
|
||||||
|
|
||||||
|
public class CheckDatabaseTest
|
||||||
|
{
|
||||||
|
[Fact]
|
||||||
|
public async Task PrintDatabaseStats()
|
||||||
|
{
|
||||||
|
var configJson = await File.ReadAllTextAsync("../../../../../src/NexusReader.Web/appsettings.json");
|
||||||
|
var doc = JsonDocument.Parse(configJson);
|
||||||
|
var pgConn = doc.RootElement.GetProperty("ConnectionStrings").GetProperty("PostgresConnection").GetString();
|
||||||
|
|
||||||
|
Console.WriteLine($"Postgres Connection: {pgConn}");
|
||||||
|
|
||||||
|
var optionsBuilder = new DbContextOptionsBuilder<AppDbContext>();
|
||||||
|
optionsBuilder.UseNpgsql(pgConn);
|
||||||
|
|
||||||
|
using var context = new AppDbContext(optionsBuilder.Options);
|
||||||
|
|
||||||
|
var usersCount = await context.Users.CountAsync();
|
||||||
|
var ebooksCount = await context.Ebooks.CountAsync();
|
||||||
|
var unitsCount = await context.KnowledgeUnits.CountAsync();
|
||||||
|
var cacheCount = await context.SemanticKnowledgeCache.CountAsync();
|
||||||
|
|
||||||
|
Console.WriteLine($"=== DATABASE STATS ===");
|
||||||
|
Console.WriteLine($"Users: {usersCount}");
|
||||||
|
Console.WriteLine($"Ebooks: {ebooksCount}");
|
||||||
|
Console.WriteLine($"KnowledgeUnits: {unitsCount}");
|
||||||
|
Console.WriteLine($"SemanticKnowledgeCache: {cacheCount}");
|
||||||
|
|
||||||
|
var users = await context.Users.ToListAsync();
|
||||||
|
foreach (var u in users)
|
||||||
|
{
|
||||||
|
Console.WriteLine($"User: {u.Email}, TenantId: '{u.TenantId}'");
|
||||||
|
}
|
||||||
|
|
||||||
|
var ebooks = await context.Ebooks.ToListAsync();
|
||||||
|
foreach (var eb in ebooks)
|
||||||
|
{
|
||||||
|
Console.WriteLine($"Ebook Id: {eb.Id}, Title: '{eb.Title}', FilePath: '{eb.FilePath}', Ready: {eb.IsReadyForReading}");
|
||||||
|
}
|
||||||
|
|
||||||
|
var cache = await context.SemanticKnowledgeCache.ToListAsync();
|
||||||
|
foreach (var c in cache)
|
||||||
|
{
|
||||||
|
Console.WriteLine($"Cache Hash: {c.ContentHash}, TenantId: '{c.TenantId}', PromptVersion: {c.PromptVersion}, JsonData Preview: {c.JsonData.Substring(0, Math.Min(c.JsonData.Length, 150))}");
|
||||||
|
}
|
||||||
|
|
||||||
|
Assert.True(true);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -116,11 +116,8 @@ public class QueryTests : IDisposable
|
|||||||
public async Task SearchLibrarySemanticallyQuery_WithEmptyQueryText_ReturnsFailure()
|
public async Task SearchLibrarySemanticallyQuery_WithEmptyQueryText_ReturnsFailure()
|
||||||
{
|
{
|
||||||
// Arrange
|
// Arrange
|
||||||
var handler = new SearchLibrarySemanticallyQueryHandler(
|
var knowledgeServiceMock = new Mock<IKnowledgeService>();
|
||||||
_embeddingGeneratorMock.Object,
|
var handler = new SearchLibrarySemanticallyQueryHandler(knowledgeServiceMock.Object);
|
||||||
_dbContextFactoryMock.Object,
|
|
||||||
_pipelineProviderMock.Object,
|
|
||||||
_mapperMock.Object);
|
|
||||||
var query = new SearchLibrarySemanticallyQuery("", "tenant-123");
|
var query = new SearchLibrarySemanticallyQuery("", "tenant-123");
|
||||||
|
|
||||||
// Act
|
// Act
|
||||||
@@ -132,39 +129,38 @@ public class QueryTests : IDisposable
|
|||||||
}
|
}
|
||||||
|
|
||||||
[Fact]
|
[Fact]
|
||||||
public async Task SearchLibrarySemanticallyQuery_WithValidQuery_GeneratesEmbeddingAndQueriesDatabase()
|
public async Task SearchLibrarySemanticallyQuery_WithValidQuery_CallsKnowledgeService()
|
||||||
{
|
{
|
||||||
// Arrange
|
// Arrange
|
||||||
var queryText = "test query";
|
var queryText = "test query";
|
||||||
var tenantId = "tenant-123";
|
var tenantId = "tenant-123";
|
||||||
|
var expectedResponse = new List<SemanticSearchResultDto>
|
||||||
|
{
|
||||||
|
new SemanticSearchResultDto
|
||||||
|
{
|
||||||
|
Snippet = "Matched content",
|
||||||
|
RelevanceScore = 0.95f,
|
||||||
|
SourceBookTitle = "Test Book"
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
var mockEmbedding = new Embedding<float>(new float[768]);
|
var knowledgeServiceMock = new Mock<IKnowledgeService>();
|
||||||
var mockResponse = new GeneratedEmbeddings<Embedding<float>>(new[] { mockEmbedding });
|
knowledgeServiceMock.Setup(s => s.SearchLibrarySemanticallyAsync(queryText, tenantId, 5, It.IsAny<CancellationToken>()))
|
||||||
_embeddingGeneratorMock.Setup(g => g.GenerateAsync(
|
.ReturnsAsync(Result.Ok(expectedResponse));
|
||||||
It.Is<IEnumerable<string>>(s => s.Contains(queryText)),
|
|
||||||
It.IsAny<EmbeddingGenerationOptions>(),
|
|
||||||
It.IsAny<CancellationToken>()))
|
|
||||||
.ReturnsAsync(mockResponse);
|
|
||||||
|
|
||||||
var handler = new SearchLibrarySemanticallyQueryHandler(
|
|
||||||
_embeddingGeneratorMock.Object,
|
|
||||||
_dbContextFactoryMock.Object,
|
|
||||||
_pipelineProviderMock.Object,
|
|
||||||
_mapperMock.Object);
|
|
||||||
|
|
||||||
|
var handler = new SearchLibrarySemanticallyQueryHandler(knowledgeServiceMock.Object);
|
||||||
var query = new SearchLibrarySemanticallyQuery(queryText, tenantId);
|
var query = new SearchLibrarySemanticallyQuery(queryText, tenantId);
|
||||||
|
|
||||||
// Act
|
// Act
|
||||||
Func<Task> act = async () => await handler.Handle(query, CancellationToken.None);
|
var result = await handler.Handle(query, CancellationToken.None);
|
||||||
|
|
||||||
// Assert (SQLite provider will throw an execution/translation exception since CosineDistance is not supported,
|
// Assert
|
||||||
// which confirms that the query built successfully and attempted execution!)
|
result.IsSuccess.Should().BeTrue();
|
||||||
await act.Should().ThrowAsync<Exception>();
|
result.Value.Should().HaveCount(1);
|
||||||
|
result.Value.First().Snippet.Should().Be("Matched content");
|
||||||
|
result.Value.First().SourceBookTitle.Should().Be("Test Book");
|
||||||
|
|
||||||
_embeddingGeneratorMock.Verify(g => g.GenerateAsync(
|
knowledgeServiceMock.Verify(s => s.SearchLibrarySemanticallyAsync(queryText, tenantId, 5, It.IsAny<CancellationToken>()), Times.Once);
|
||||||
It.Is<IEnumerable<string>>(s => s.Contains(queryText)),
|
|
||||||
It.IsAny<EmbeddingGenerationOptions>(),
|
|
||||||
It.IsAny<CancellationToken>()), Times.Once);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
[Fact]
|
[Fact]
|
||||||
|
|||||||
Reference in New Issue
Block a user