541e9e1fb5
This Pull Request encapsulates all outstanding AI, Blazor InteractiveAuto lifecycle, pgvector, and Firefox authorization/session compatibility fixes. ### Key Accomplishments: 1. **Concurrent Request Deduplication (Option B):** Implemented a thread-safe active task registry in `KnowledgeService` that groups concurrent graph extraction queries for the same content, preventing duplicate AI calls completely. 2. **Resilience Strategy for Downstream Demands:** Extended the `ai-retry` resilience pipeline to automatically intercept and retry on temporary Google API `503 ServiceUnavailable` / `high demand` spikes. 3. **Interactive Graph Generation Guard (Option A):** Prevented server-side prerender-phase graph requests in the reader canvas component. 4. **Firefox Compatibility & Cookie Handler:** Implemented an authentication endpoint and hybrid hidden-form submission flow to solve login, registration, and logout redirections and cookies securely. 5. **Autoscrolling & Graph Exclusions:** Added concept-to-block smooth scrolling, active block badging, and filtered out markdown code blocks from being extracted as nodes. All unit tests compiled and passed 100% cleanly. --------- Co-authored-by: Marek Jasiński <jasins.marek@gmail.com> Reviewed-on: #44 Co-authored-by: Antigravity <antigravity@google.com> Co-committed-by: Antigravity <antigravity@google.com>
68 lines
2.8 KiB
C#
68 lines
2.8 KiB
C#
using MediatR;
|
|
using FluentResults;
|
|
using Microsoft.EntityFrameworkCore;
|
|
using NexusReader.Application.DTOs.User;
|
|
using NexusReader.Data.Persistence;
|
|
|
|
namespace NexusReader.Application.Queries.User;
|
|
|
|
public class GetUserProfileQueryHandler : IRequestHandler<GetUserProfileQuery, Result<UserProfileDto>>
|
|
{
|
|
private readonly IDbContextFactory<AppDbContext> _dbContextFactory;
|
|
|
|
public GetUserProfileQueryHandler(IDbContextFactory<AppDbContext> dbContextFactory)
|
|
{
|
|
_dbContextFactory = dbContextFactory;
|
|
}
|
|
|
|
public async Task<Result<UserProfileDto>> Handle(GetUserProfileQuery request, CancellationToken cancellationToken)
|
|
{
|
|
using var dbContext = await _dbContextFactory.CreateDbContextAsync(cancellationToken);
|
|
var profile = await dbContext.Users
|
|
.Where(u => u.Id == request.UserId)
|
|
.Select(u => new UserProfileDto
|
|
{
|
|
Email = u.Email ?? string.Empty,
|
|
AITokensUsed = u.AITokensUsed,
|
|
TenantId = u.TenantId != null && u.TenantId.Length == 36 ? new Guid(u.TenantId) : Guid.Empty,
|
|
Plan = u.SubscriptionPlan != null ? new SubscriptionPlanDto
|
|
{
|
|
Id = u.SubscriptionPlan.Id,
|
|
Name = u.SubscriptionPlan.PlanName,
|
|
AITokenLimit = u.SubscriptionPlan.AITokenLimit,
|
|
MonthlyPrice = u.SubscriptionPlan.MonthlyPrice
|
|
} : new SubscriptionPlanDto(),
|
|
AverageQuizScore = u.QuizResults.Any(q => q.TotalQuestions > 0)
|
|
? (int)u.QuizResults.Where(q => q.TotalQuestions > 0).Average(q => (double)q.Score / q.TotalQuestions * 100)
|
|
: 0,
|
|
LastReadBook = u.Ebooks.OrderByDescending(e => e.LastReadDate).Select(e => new LastReadBookDto
|
|
{
|
|
Id = e.Id,
|
|
Title = e.Title,
|
|
Author = new AuthorDto
|
|
{
|
|
Id = e.Author.Id,
|
|
Name = e.Author.Name
|
|
},
|
|
CoverUrl = e.CoverUrl,
|
|
Progress = e.Progress,
|
|
LastChapter = e.LastChapter ?? "Rozpoczynanie...",
|
|
LastChapterIndex = e.LastChapterIndex,
|
|
Description = e.Description
|
|
}).FirstOrDefault(),
|
|
Roles = dbContext.UserRoles
|
|
.Where(ur => ur.UserId == u.Id)
|
|
.Join(dbContext.Roles, ur => ur.RoleId, r => r.Id, (ur, r) => r.Name!)
|
|
.ToArray()
|
|
})
|
|
.FirstOrDefaultAsync(cancellationToken);
|
|
|
|
if (profile == null)
|
|
{
|
|
return Result.Fail("Profile not found.");
|
|
}
|
|
|
|
return Result.Ok(profile);
|
|
}
|
|
}
|