Iceshrimp.NET/Iceshrimp.Backend/Core/Database/Tables/User.cs

718 lines
27 KiB
C#

using System.ComponentModel.DataAnnotations;
using System.ComponentModel.DataAnnotations.Schema;
using EntityFrameworkCore.Projectables;
using Iceshrimp.Backend.Core.Configuration;
using Iceshrimp.Backend.Core.Helpers;
using Iceshrimp.Shared.Helpers;
using Microsoft.EntityFrameworkCore;
using Microsoft.EntityFrameworkCore.Metadata.Builders;
namespace Iceshrimp.Backend.Core.Database.Tables;
[Table("user")]
[Index(nameof(Host))]
[Index(nameof(UsernameLower), nameof(Host), IsUnique = true)]
[Index(nameof(UpdatedAt))]
[Index(nameof(UsernameLower))]
[Index(nameof(Uri))]
[Index(nameof(LastActiveDate))]
[Index(nameof(IsExplorable))]
[Index(nameof(IsAdmin))]
[Index(nameof(IsModerator))]
[Index(nameof(CreatedAt))]
[Index(nameof(Tags))]
[Index(nameof(AvatarId), IsUnique = true)]
[Index(nameof(BannerId), IsUnique = true)]
[Index(nameof(IsSuspended))]
public class User : IIdentifiable
{
/// <summary>
/// The created date of the User.
/// </summary>
[Column("createdAt")]
public DateTime CreatedAt { get; set; }
/// <summary>
/// The updated date of the User.
/// </summary>
[Column("updatedAt")]
public DateTime? UpdatedAt { get; set; }
[Column("lastFetchedAt")] public DateTime? LastFetchedAt { get; set; }
[Column("outboxFetchedAt")] public DateTime? OutboxFetchedAt { get; set; }
[NotMapped]
[Projectable]
public bool NeedsUpdate =>
Host != null && (LastFetchedAt == null || LastFetchedAt < DateTime.Now - TimeSpan.FromHours(24));
/// <summary>
/// The username of the User.
/// </summary>
[Column("username")]
[StringLength(128)]
public string Username { get; set; } = null!;
/// <summary>
/// The username (lowercased) of the User.
/// </summary>
[Column("usernameLower")]
[StringLength(128)]
public string UsernameLower { get; set; } = null!;
/// <summary>
/// The name of the User.
/// </summary>
[Column("name")]
[StringLength(128)]
public string? DisplayName { get; set; }
/// <summary>
/// The count of followers.
/// </summary>
[Column("followersCount")]
public int FollowersCount { get; set; }
/// <summary>
/// The count of following.
/// </summary>
[Column("followingCount")]
public int FollowingCount { get; set; }
/// <summary>
/// The count of notes.
/// </summary>
[Column("notesCount")]
public int NotesCount { get; set; }
/// <summary>
/// The ID of avatar DriveFile.
/// </summary>
[Column("avatarId")]
[StringLength(32)]
public string? AvatarId { get; set; }
/// <summary>
/// The ID of banner DriveFile.
/// </summary>
[Column("bannerId")]
[StringLength(32)]
public string? BannerId { get; set; }
[Column("tags", TypeName = "character varying(128)[]")]
public List<string> Tags { get; set; } = [];
/// <summary>
/// Whether the User is suspended.
/// </summary>
[Column("isSuspended")]
public bool IsSuspended { get; set; }
/// <summary>
/// Whether the User is silenced.
/// </summary>
[Column("isSilenced")]
public bool IsSilenced { get; set; }
/// <summary>
/// Whether the User is locked.
/// </summary>
[Column("isLocked")]
public bool IsLocked { get; set; }
/// <summary>
/// Whether the User is a bot.
/// </summary>
[Column("isBot")]
public bool IsBot { get; set; }
[Column("isSystem")] public bool IsSystemUser { get; set; }
[Column("isRelayActor")] public bool IsRelayActor { get; set; }
/// <summary>
/// Whether the User is a cat.
/// </summary>
[Column("isCat")]
public bool IsCat { get; set; }
/// <summary>
/// Whether the User is the admin.
/// </summary>
[Column("isAdmin")]
public bool IsAdmin { get; set; }
/// <summary>
/// Whether the User is a moderator.
/// </summary>
[Column("isModerator")]
public bool IsModerator { get; set; }
[Column("emojis", TypeName = "character varying(128)[]")]
public List<string> Emojis { get; set; } = [];
/// <summary>
/// The host of the User. It will be null if the origin of the user is local.
/// </summary>
[Column("host")]
[StringLength(512)]
public string? Host { get; set; }
[NotMapped] [Projectable] public string Acct => Username + (Host != null ? "@" + Host : "");
[NotMapped] [Projectable] public string AcctWithPrefix => "acct:" + Acct;
/// <summary>
/// The inbox URL of the User. It will be null if the origin of the user is local.
/// </summary>
[Column("inbox")]
[StringLength(512)]
public string? Inbox { get; set; }
/// <summary>
/// The outbox URL of the User. It will be null if the origin of the user is local, or if no outbox is present.
/// </summary>
[Column("outbox")]
[StringLength(512)]
public string? Outbox { get; set; }
/// <summary>
/// The sharedInbox URL of the User. It will be null if the origin of the user is local.
/// </summary>
[Column("sharedInbox")]
[StringLength(512)]
public string? SharedInbox { get; set; }
/// <summary>
/// The featured URL of the User. It will be null if the origin of the user is local.
/// </summary>
[Column("featured")]
[StringLength(512)]
public string? Featured { get; set; }
/// <summary>
/// The URI of the User. It will be null if the origin of the user is local.
/// </summary>
[Column("uri")]
[StringLength(512)]
public string? Uri { get; set; }
/// <summary>
/// Whether the User is explorable.
/// </summary>
[Column("isExplorable")]
public bool IsExplorable { get; set; }
/// <summary>
/// The URI of the user Follower Collection. It will be null if the origin of the user is local.
/// </summary>
[Column("followersUri")]
[StringLength(512)]
public string? FollowersUri { get; set; }
[Column("lastActiveDate")] public DateTime? LastActiveDate { get; set; }
/// <summary>
/// Whether the User is deleted.
/// </summary>
[Column("isDeleted")]
public bool IsDeleted { get; set; }
/// <summary>
/// Overrides user drive capacity limit
/// </summary>
[Column("driveCapacityOverrideMb")]
public int? DriveCapacityOverrideMb { get; set; }
/// <summary>
/// The URI of the new account of the User
/// </summary>
[Column("movedToUri")]
[StringLength(512)]
public string? MovedToUri { get; set; }
/// <summary>
/// URIs the user is known as too
/// </summary>
[Column("alsoKnownAs")]
public List<string>? AlsoKnownAs { get; set; }
/// <summary>
/// Whether to speak as a cat if isCat.
/// </summary>
[Column("speakAsCat")]
public bool SpeakAsCat { get; set; }
/// <summary>
/// The blurhash of the avatar DriveFile
/// </summary>
[Column("avatarBlurhash")]
[StringLength(128)]
public string? AvatarBlurhash { get; set; }
/// <summary>
/// The blurhash of the banner DriveFile
/// </summary>
[Column("bannerBlurhash")]
[StringLength(128)]
public string? BannerBlurhash { get; set; }
[Column("splitDomainResolved")] public bool SplitDomainResolved { get; set; }
[InverseProperty(nameof(AbuseUserReport.Assignee))]
public virtual ICollection<AbuseUserReport> AbuseUserReportAssignees { get; set; } = new List<AbuseUserReport>();
[InverseProperty(nameof(AbuseUserReport.Reporter))]
public virtual ICollection<AbuseUserReport> AbuseUserReportReporters { get; set; } = new List<AbuseUserReport>();
[InverseProperty(nameof(AbuseUserReport.TargetUser))]
public virtual ICollection<AbuseUserReport> AbuseUserReportTargetUsers { get; set; } = new List<AbuseUserReport>();
[InverseProperty(nameof(AnnouncementRead.User))]
public virtual ICollection<AnnouncementRead> AnnouncementReads { get; set; } = new List<AnnouncementRead>();
[InverseProperty(nameof(Antenna.User))]
public virtual ICollection<Antenna> Antennas { get; set; } = new List<Antenna>();
[InverseProperty(nameof(AttestationChallenge.User))]
public virtual ICollection<AttestationChallenge> AttestationChallenges { get; set; } =
new List<AttestationChallenge>();
[ForeignKey(nameof(AvatarId))]
[InverseProperty(nameof(DriveFile.UserAvatar))]
public virtual DriveFile? Avatar { get; set; }
[ForeignKey(nameof(BannerId))]
[InverseProperty(nameof(DriveFile.UserBanner))]
public virtual DriveFile? Banner { get; set; }
[InverseProperty(nameof(Tables.Blocking.Blockee))]
public virtual ICollection<Blocking> IncomingBlocks { get; set; } = new List<Blocking>();
[InverseProperty(nameof(Tables.Blocking.Blocker))]
public virtual ICollection<Blocking> OutgoingBlocks { get; set; } = new List<Blocking>();
[NotMapped] [Projectable] public virtual IEnumerable<User> BlockedBy => IncomingBlocks.Select(p => p.Blocker);
[NotMapped] [Projectable] public virtual IEnumerable<User> Blocking => OutgoingBlocks.Select(p => p.Blockee);
[InverseProperty(nameof(ChannelFollowing.Follower))]
public virtual ICollection<ChannelFollowing> ChannelFollowings { get; set; } = new List<ChannelFollowing>();
[InverseProperty(nameof(Channel.User))]
public virtual ICollection<Channel> Channels { get; set; } = new List<Channel>();
[InverseProperty(nameof(Clip.User))] public virtual ICollection<Clip> Clips { get; set; } = new List<Clip>();
[InverseProperty(nameof(DriveFile.User))]
public virtual ICollection<DriveFile> DriveFiles { get; set; } = new List<DriveFile>();
[InverseProperty(nameof(DriveFolder.User))]
public virtual ICollection<DriveFolder> DriveFolders { get; set; } = new List<DriveFolder>();
[InverseProperty(nameof(FollowRequest.Followee))]
public virtual ICollection<FollowRequest> IncomingFollowRequests { get; set; } = new List<FollowRequest>();
[InverseProperty(nameof(FollowRequest.Follower))]
public virtual ICollection<FollowRequest> OutgoingFollowRequests { get; set; } = new List<FollowRequest>();
[NotMapped]
[Projectable]
public virtual IEnumerable<User> ReceivedFollowRequests => IncomingFollowRequests.Select(p => p.Follower);
[NotMapped]
[Projectable]
public virtual IEnumerable<User> SentFollowRequests => OutgoingFollowRequests.Select(p => p.Followee);
[InverseProperty(nameof(Tables.Following.Followee))]
public virtual ICollection<Following> IncomingFollowRelationships { get; set; } = new List<Following>();
[InverseProperty(nameof(Tables.Following.Follower))]
public virtual ICollection<Following> OutgoingFollowRelationships { get; set; } = new List<Following>();
[NotMapped]
[Projectable]
public virtual IEnumerable<User> Followers => IncomingFollowRelationships.Select(p => p.Follower);
[NotMapped]
[Projectable]
public virtual IEnumerable<User> Following => OutgoingFollowRelationships.Select(p => p.Followee);
[InverseProperty(nameof(GalleryLike.User))]
public virtual ICollection<GalleryLike> GalleryLikes { get; set; } = new List<GalleryLike>();
[InverseProperty(nameof(GalleryPost.User))]
public virtual ICollection<GalleryPost> GalleryPosts { get; set; } = new List<GalleryPost>();
[InverseProperty(nameof(Marker.User))]
public virtual ICollection<Marker> Markers { get; set; } = new List<Marker>();
[InverseProperty(nameof(MessagingMessage.Recipient))]
public virtual ICollection<MessagingMessage> MessagingMessageRecipients { get; set; } =
new List<MessagingMessage>();
[InverseProperty(nameof(MessagingMessage.User))]
public virtual ICollection<MessagingMessage> MessagingMessageUsers { get; set; } = new List<MessagingMessage>();
[InverseProperty(nameof(ModerationLog.User))]
public virtual ICollection<ModerationLog> ModerationLogs { get; set; } = new List<ModerationLog>();
[InverseProperty(nameof(Tables.Muting.Mutee))]
public virtual ICollection<Muting> IncomingMutes { get; set; } = new List<Muting>();
[InverseProperty(nameof(Tables.Muting.Muter))]
public virtual ICollection<Muting> OutgoingMutes { get; set; } = new List<Muting>();
[NotMapped] [Projectable] public virtual IEnumerable<User> MutedBy => IncomingMutes.Select(p => p.Muter);
[NotMapped] [Projectable] public virtual IEnumerable<User> Muting => OutgoingMutes.Select(p => p.Mutee);
[InverseProperty(nameof(NoteBookmark.User))]
public virtual ICollection<NoteBookmark> NoteBookmarks { get; set; } = new List<NoteBookmark>();
[NotMapped] [Projectable] public virtual IEnumerable<Note> BookmarkedNotes => NoteBookmarks.Select(p => p.Note);
[InverseProperty(nameof(NoteReaction.User))]
public virtual ICollection<NoteLike> NoteLikes { get; set; } = new List<NoteLike>();
[NotMapped] [Projectable] public virtual IEnumerable<Note> LikedNotes => NoteLikes.Select(p => p.Note);
[InverseProperty(nameof(NoteReaction.User))]
public virtual ICollection<NoteReaction> NoteReactions { get; set; } = new List<NoteReaction>();
[NotMapped]
[Projectable]
public virtual IEnumerable<Note> ReactedNotes => NoteReactions.Select(p => p.Note).Distinct();
[InverseProperty(nameof(NoteThreadMuting.User))]
public virtual ICollection<NoteThreadMuting> NoteThreadMutings { get; set; } = new List<NoteThreadMuting>();
[InverseProperty(nameof(NoteUnread.User))]
public virtual ICollection<NoteUnread> NoteUnreads { get; set; } = new List<NoteUnread>();
[InverseProperty(nameof(NoteWatching.User))]
public virtual ICollection<NoteWatching> NoteWatchings { get; set; } = new List<NoteWatching>();
[InverseProperty(nameof(Note.User))] public virtual ICollection<Note> Notes { get; set; } = new List<Note>();
[InverseProperty(nameof(Notification.Notifiee))]
public virtual ICollection<Notification> NotificationNotifiees { get; set; } = new List<Notification>();
[InverseProperty(nameof(Notification.Notifier))]
public virtual ICollection<Notification> NotificationNotifiers { get; set; } = new List<Notification>();
[InverseProperty(nameof(OauthToken.User))]
public virtual ICollection<OauthToken> OauthTokens { get; set; } = new List<OauthToken>();
[InverseProperty(nameof(PageLike.User))]
public virtual ICollection<PageLike> PageLikes { get; set; } = new List<PageLike>();
[InverseProperty(nameof(Page.User))] public virtual ICollection<Page> Pages { get; set; } = new List<Page>();
[InverseProperty(nameof(PasswordResetRequest.User))]
public virtual ICollection<PasswordResetRequest> PasswordResetRequests { get; set; } =
new List<PasswordResetRequest>();
[InverseProperty(nameof(PollVote.User))]
public virtual ICollection<PollVote> PollVotes { get; set; } = new List<PollVote>();
[InverseProperty(nameof(PromoRead.User))]
public virtual ICollection<PromoRead> PromoReads { get; set; } = new List<PromoRead>();
[InverseProperty(nameof(RegistryItem.User))]
public virtual ICollection<RegistryItem> RegistryItems { get; set; } = new List<RegistryItem>();
[InverseProperty(nameof(RenoteMuting.Mutee))]
public virtual ICollection<RenoteMuting> RenoteMutingMutees { get; set; } = new List<RenoteMuting>();
[InverseProperty(nameof(RenoteMuting.Muter))]
public virtual ICollection<RenoteMuting> RenoteMutingMuters { get; set; } = new List<RenoteMuting>();
[InverseProperty(nameof(Session.User))]
public virtual ICollection<Session> Sessions { get; set; } = new List<Session>();
[InverseProperty(nameof(SwSubscription.User))]
public virtual ICollection<SwSubscription> SwSubscriptions { get; set; } = new List<SwSubscription>();
[InverseProperty(nameof(PushSubscription.User))]
public virtual ICollection<PushSubscription> PushSubscriptions { get; set; } = new List<PushSubscription>();
[InverseProperty(nameof(UserGroupInvitation.User))]
public virtual ICollection<UserGroupInvitation> UserGroupInvitations { get; set; } =
new List<UserGroupInvitation>();
[InverseProperty(nameof(UserGroupMember.User))]
public virtual ICollection<UserGroupMember> UserGroupMemberships { get; set; } = new List<UserGroupMember>();
[InverseProperty(nameof(UserGroup.User))]
public virtual ICollection<UserGroup> UserGroups { get; set; } = new List<UserGroup>();
[InverseProperty(nameof(Tables.UserKeypair.User))]
public virtual UserKeypair? UserKeypair { get; set; }
[InverseProperty(nameof(UserListMember.User))]
public virtual ICollection<UserListMember> UserListMembers { get; set; } = new List<UserListMember>();
[InverseProperty(nameof(UserList.User))]
public virtual ICollection<UserList> UserLists { get; set; } = new List<UserList>();
[InverseProperty(nameof(UserNotePin.User))]
public virtual ICollection<UserNotePin> UserNotePins { get; set; } = new List<UserNotePin>();
[NotMapped] [Projectable] public virtual IEnumerable<Note> PinnedNotes => UserNotePins.Select(p => p.Note);
[InverseProperty(nameof(Tables.UserProfile.User))]
public virtual UserProfile? UserProfile { get; set; }
[InverseProperty(nameof(Tables.UserPublickey.User))]
public virtual UserPublickey? UserPublickey { get; set; }
[InverseProperty(nameof(Tables.UserSettings.User))]
public virtual UserSettings? UserSettings { get; set; }
[InverseProperty(nameof(UserSecurityKey.User))]
public virtual ICollection<UserSecurityKey> UserSecurityKeys { get; set; } = new List<UserSecurityKey>();
[InverseProperty(nameof(Webhook.User))]
public virtual ICollection<Webhook> Webhooks { get; set; } = new List<Webhook>();
[InverseProperty(nameof(Filter.User))]
public virtual ICollection<Filter> Filters { get; set; } = new List<Filter>();
[NotMapped] public bool? PrecomputedIsBlocking { get; set; }
[NotMapped] public bool? PrecomputedIsBlockedBy { get; set; }
[NotMapped] public bool? PrecomputedIsMuting { get; set; }
[NotMapped] public bool? PrecomputedIsMutedBy { get; set; }
[NotMapped] public bool? PrecomputedIsFollowing { get; set; }
[NotMapped] public bool? PrecomputedIsFollowedBy { get; set; }
[NotMapped] public bool? PrecomputedIsRequested { get; set; }
[NotMapped] public bool? PrecomputedIsRequestedBy { get; set; }
[Projectable] public bool IsLocalUser => Host == null;
[Projectable] public bool IsRemoteUser => Host != null;
[Projectable] public string IdenticonUrlPath => $"/identicon/{Id}";
[Key]
[Column("id")]
[StringLength(32)]
public string Id { get; set; } = null!;
[Projectable]
public string GetFqnLower(string accountDomain) => UsernameLower + "@" + (Host ?? accountDomain);
[Projectable]
public string GetFqn(string accountDomain) => Username + "@" + (Host ?? accountDomain);
[Projectable]
public bool DisplayNameContainsCaseInsensitive(string str) =>
DisplayName != null && EF.Functions.ILike(DisplayName, "%" + EfHelpers.EscapeLikeQuery(str) + "%", @"\");
[Projectable]
public bool UsernameContainsCaseInsensitive(string str) => UsernameLower.Contains(str.ToLowerInvariant());
[Projectable]
public bool FqnContainsCaseInsensitive(string str, string accountDomain) =>
GetFqnLower(accountDomain).Contains(str.ToLowerInvariant());
[Projectable]
public bool UsernameOrFqnContainsCaseInsensitive(string str, string accountDomain) =>
str.Contains('@') ? FqnContainsCaseInsensitive(str, accountDomain) : UsernameContainsCaseInsensitive(str);
[Projectable]
public bool DisplayNameOrUsernameOrFqnContainsCaseInsensitive(string str, string accountDomain) =>
str.Contains('@') && !str.Contains(' ')
? FqnContainsCaseInsensitive(str, accountDomain)
: UsernameContainsCaseInsensitive(str) || DisplayNameContainsCaseInsensitive(str);
[Projectable]
public bool IsBlockedBy(User user) => BlockedBy.Contains(user);
[Projectable]
public bool IsBlocking(User user) => Blocking.Contains(user);
[Projectable]
public bool IsFollowedBy(User user) => Followers.Contains(user);
[Projectable]
public bool IsFollowing(User user) => Following.Contains(user);
[Projectable]
public bool IsRequestedBy(User user) => ReceivedFollowRequests.Contains(user);
[Projectable]
public bool IsRequested(User user) => SentFollowRequests.Contains(user);
[Projectable]
public bool IsMutedBy(User user) => MutedBy.Contains(user);
[Projectable]
public bool IsMuting(User user) => Muting.Contains(user);
[Projectable]
public bool HasPinned(Note note) => PinnedNotes.Contains(note);
[Projectable]
public bool HasBookmarked(Note note) => BookmarkedNotes.Contains(note);
[Projectable]
public bool HasLiked(Note note) => LikedNotes.Contains(note);
[Projectable]
public bool HasReacted(Note note) => ReactedNotes.Contains(note);
[Projectable]
public bool HasRenoted(Note note) => Notes.Any(p => p.Renote == note && p.User == this);
[Projectable]
public bool HasReplied(Note note) => Notes.Any(p => p.Reply == note && p.User == this);
[Projectable]
public bool HasVoted(Note note) => PollVotes.Any(p => p.Note == note && p.User == this);
[Projectable]
public bool HasInteractedWith(Note note) =>
HasLiked(note)
|| HasReacted(note)
|| HasBookmarked(note)
|| HasReplied(note)
|| HasRenoted(note)
|| HasVoted(note);
[Projectable]
public bool ProhibitInteractionWith(User user) => IsBlocking(user) || IsBlockedBy(user);
public User WithPrecomputedBlockStatus(bool blocking, bool blockedBy)
{
PrecomputedIsBlocking = blocking;
PrecomputedIsBlockedBy = blockedBy;
return this;
}
public User WithPrecomputedMuteStatus(bool muting, bool mutedBy)
{
PrecomputedIsMuting = muting;
PrecomputedIsMutedBy = mutedBy;
return this;
}
public User WithPrecomputedFollowStatus(bool following, bool followedBy, bool requested, bool requestedBy)
{
PrecomputedIsFollowing = following;
PrecomputedIsFollowedBy = followedBy;
PrecomputedIsRequested = requested;
PrecomputedIsRequestedBy = requestedBy;
return this;
}
public string GetPublicUrl(Config.InstanceSection config) => GetPublicUrl(config.WebDomain);
public string GetPublicUri(Config.InstanceSection config) => GetPublicUri(config.WebDomain);
public string GetUriOrPublicUri(Config.InstanceSection config) => GetUriOrPublicUri(config.WebDomain);
public string GetIdenticonUrl(Config.InstanceSection config) => GetIdenticonUrl(config.WebDomain);
public string GetPublicUri(string webDomain) => Host == null
? $"https://{webDomain}/users/{Id}"
: throw new Exception("Cannot access PublicUri for remote user");
public string GetUriOrPublicUri(string webDomain) => Uri ?? GetPublicUri(webDomain);
public string GetPublicUrl(string webDomain) => Host == null
? $"https://{webDomain}{PublicUrlPath}"
: throw new Exception("Cannot access PublicUrl for remote user");
[Projectable] public string PublicUrlPath => $"/@{Username}";
public string GetIdenticonUrl(string webDomain) => $"https://{webDomain}{IdenticonUrlPath}";
public string GetAvatarUrl(Config.InstanceSection config)
=> $"https://{config.WebDomain}/avatars/{Id}/{AvatarId ?? "identicon"}";
public string? GetBannerUrl(Config.InstanceSection config)
=> BannerId != null ? $"https://{config.WebDomain}/banners/{Id}/{BannerId}" : null;
private class EntityTypeConfiguration : IEntityTypeConfiguration<User>
{
public void Configure(EntityTypeBuilder<User> entity)
{
entity.Property(e => e.AlsoKnownAs).HasComment("URIs the user is known as too");
entity.Property(e => e.AvatarBlurhash).HasComment("The blurhash of the avatar DriveFile");
entity.Property(e => e.AvatarId).HasComment("The ID of avatar DriveFile.");
entity.Property(e => e.BannerBlurhash).HasComment("The blurhash of the banner DriveFile");
entity.Property(e => e.BannerId).HasComment("The ID of banner DriveFile.");
entity.Property(e => e.CreatedAt).HasComment("The created date of the User.");
entity.Property(e => e.DriveCapacityOverrideMb).HasComment("Overrides user drive capacity limit");
entity.Property(e => e.Emojis).HasDefaultValueSql("'{}'::character varying[]");
entity.Property(e => e.Featured)
.HasComment("The featured URL of the User. It will be null if the origin of the user is local.");
entity.Property(e => e.FollowersCount)
.HasDefaultValue(0)
.HasComment("The count of followers.");
entity.Property(e => e.FollowersUri)
.HasComment("The URI of the user Follower Collection. It will be null if the origin of the user is local.");
entity.Property(e => e.FollowingCount)
.HasDefaultValue(0)
.HasComment("The count of following.");
entity.Property(e => e.Host)
.HasComment("The host of the User. It will be null if the origin of the user is local.");
entity.Property(e => e.Inbox)
.HasComment("The inbox URL of the User. It will be null if the origin of the user is local.");
entity.Property(e => e.IsAdmin)
.HasDefaultValue(false)
.HasComment("Whether the User is the admin.");
entity.Property(e => e.IsBot)
.HasDefaultValue(false)
.HasComment("Whether the User is a bot.");
entity.Property(e => e.IsCat)
.HasDefaultValue(false)
.HasComment("Whether the User is a cat.");
entity.Property(e => e.IsDeleted)
.HasDefaultValue(false)
.HasComment("Whether the User is deleted.");
entity.Property(e => e.IsExplorable)
.HasDefaultValue(true)
.HasComment("Whether the User is explorable.");
entity.Property(e => e.IsLocked)
.HasDefaultValue(false)
.HasComment("Whether the User is locked.");
entity.Property(e => e.IsModerator)
.HasDefaultValue(false)
.HasComment("Whether the User is a moderator.");
entity.Property(e => e.IsSilenced)
.HasDefaultValue(false)
.HasComment("Whether the User is silenced.");
entity.Property(e => e.IsSuspended)
.HasDefaultValue(false)
.HasComment("Whether the User is suspended.");
entity.Property(e => e.MovedToUri).HasComment("The URI of the new account of the User");
entity.Property(e => e.DisplayName).HasComment("The name of the User.");
entity.Property(e => e.NotesCount)
.HasDefaultValue(0)
.HasComment("The count of notes.");
entity.Property(e => e.SharedInbox)
.HasComment("The sharedInbox URL of the User. It will be null if the origin of the user is local.");
entity.Property(e => e.SpeakAsCat)
.HasDefaultValue(true)
.HasComment("Whether to speak as a cat if isCat.");
entity.Property(e => e.Tags).HasDefaultValueSql("'{}'::character varying[]");
entity.Property(e => e.UpdatedAt).HasComment("The updated date of the User.");
entity.Property(e => e.Uri)
.HasComment("The URI of the User. It will be null if the origin of the user is local.");
entity.Property(e => e.Username).HasComment("The username of the User.");
entity.Property(e => e.UsernameLower).HasComment("The username (lowercased) of the User.");
entity.Property(e => e.SplitDomainResolved).HasDefaultValue(false);
entity.HasOne(d => d.Avatar)
.WithOne(p => p.UserAvatar)
.OnDelete(DeleteBehavior.SetNull);
entity.HasOne(d => d.Banner)
.WithOne(p => p.UserBanner)
.OnDelete(DeleteBehavior.SetNull);
}
}
}