using Foxnouns.Backend.Database; using Foxnouns.Backend.Database.Models; using Foxnouns.Backend.Utils; using Microsoft.EntityFrameworkCore; using Newtonsoft.Json; using NodaTime; using Org.BouncyCastle.Ocsp; namespace Foxnouns.Backend.Services; public class UserRendererService( DatabaseContext db, MemberRendererService memberRenderer, Config config ) { public async Task RenderUserAsync( User user, User? selfUser = null, Token? token = null, bool renderMembers = true, bool renderAuthMethods = false, string? overrideSid = null, CancellationToken ct = default ) => await RenderUserInnerAsync( user, selfUser != null && selfUser.Id == user.Id, token?.Scopes ?? [], renderMembers, renderAuthMethods, overrideSid, ct ); public async Task RenderUserInnerAsync( User user, bool isSelfUser, string[] scopes, bool renderMembers = true, bool renderAuthMethods = false, string? overrideSid = null, CancellationToken ct = default ) { scopes = scopes.ExpandScopes(); var tokenCanReadHiddenMembers = scopes.Contains("member.read") && isSelfUser; var tokenHidden = scopes.Contains("user.read_hidden") && isSelfUser; var tokenPrivileged = scopes.Contains("user.read_privileged") && isSelfUser; renderMembers = renderMembers && (!user.ListHidden || tokenCanReadHiddenMembers); renderAuthMethods = renderAuthMethods && tokenPrivileged; IEnumerable members = renderMembers ? await db.Members.Where(m => m.UserId == user.Id).OrderBy(m => m.Name).ToListAsync(ct) : []; // Unless the user is requesting their own members AND the token can read hidden members, we filter out unlisted members. if (!(isSelfUser && tokenCanReadHiddenMembers)) members = members.Where(m => !m.Unlisted); var flags = await db .UserFlags.Where(f => f.UserId == user.Id) .OrderBy(f => f.Id) .ToListAsync(ct); var authMethods = renderAuthMethods ? await db .AuthMethods.Where(a => a.UserId == user.Id) .Include(a => a.FediverseApplication) .ToListAsync(ct) : []; int? utcOffset = null; if ( user.Timezone != null && TimeZoneInfo.TryFindSystemTimeZoneById(user.Timezone, out var tz) ) utcOffset = (int)tz.GetUtcOffset(DateTimeOffset.UtcNow).TotalSeconds; return new UserResponse( user.Id, overrideSid ?? user.Sid, user.Username, user.DisplayName, user.Bio, user.MemberTitle, AvatarUrlFor(user), user.Links, user.Names, user.Pronouns, user.Fields, user.CustomPreferences, flags.Select(f => RenderPrideFlag(f.PrideFlag)), utcOffset, user.Role, renderMembers ? members.Select(m => memberRenderer.RenderPartialMember(m, tokenHidden)) : null, renderAuthMethods ? authMethods.Select(RenderAuthMethod) : null, tokenHidden ? user.ListHidden : null, tokenHidden ? user.LastActive : null, tokenHidden ? user.LastSidReroll : null, tokenHidden ? user.Timezone ?? "" : null ); } public static AuthMethodResponse RenderAuthMethod(AuthMethod a) => new( a.Id, a.AuthType, a.RemoteId, a.FediverseApplication != null ? $"@{a.RemoteUsername}@{a.FediverseApplication.Domain}" : a.RemoteUsername ); public PartialUser RenderPartialUser(User user) => new( user.Id, user.Sid, user.Username, user.DisplayName, AvatarUrlFor(user), user.CustomPreferences ); public string? AvatarUrlFor(User user) => user.Avatar != null ? $"{config.MediaBaseUrl}/users/{user.Id}/avatars/{user.Avatar}.webp" : null; public string ImageUrlFor(PrideFlag flag) => $"{config.MediaBaseUrl}/flags/{flag.Hash}.webp"; public record UserResponse( Snowflake Id, string Sid, string Username, string? DisplayName, string? Bio, string? MemberTitle, string? AvatarUrl, string[] Links, IEnumerable Names, IEnumerable Pronouns, IEnumerable Fields, Dictionary CustomPreferences, IEnumerable Flags, int? UtcOffset, [property: JsonConverter(typeof(ScreamingSnakeCaseEnumConverter))] UserRole Role, [property: JsonProperty(NullValueHandling = NullValueHandling.Ignore)] IEnumerable? Members, [property: JsonProperty(NullValueHandling = NullValueHandling.Ignore)] IEnumerable? AuthMethods, [property: JsonProperty(NullValueHandling = NullValueHandling.Ignore)] bool? MemberListHidden, [property: JsonProperty(NullValueHandling = NullValueHandling.Ignore)] Instant? LastActive, [property: JsonProperty(NullValueHandling = NullValueHandling.Ignore)] Instant? LastSidReroll, [property: JsonProperty(NullValueHandling = NullValueHandling.Ignore)] string? Timezone ); public record AuthMethodResponse( Snowflake Id, [property: JsonConverter(typeof(ScreamingSnakeCaseEnumConverter))] AuthType Type, string RemoteId, [property: JsonProperty(NullValueHandling = NullValueHandling.Ignore)] string? RemoteUsername ); public record PartialUser( Snowflake Id, string Sid, string Username, string? DisplayName, string? AvatarUrl, Dictionary CustomPreferences ); public PrideFlagResponse RenderPrideFlag(PrideFlag flag) => new(flag.Id, ImageUrlFor(flag), flag.Name, flag.Description); public record PrideFlagResponse( Snowflake Id, string ImageUrl, string Name, string? Description ); }