Iceshrimp.NET/Iceshrimp.Backend/Core/Services/SystemUserService.cs
Laura Hausmann 09cda1a89e
[backend/database] Move user_profile columns that only concern local users to user_settings
This commit also removes a bunch of obsolete user_profile columns.
2024-07-10 02:55:57 +02:00

102 lines
No EOL
3.3 KiB
C#

using System.Security.Cryptography;
using AsyncKeyedLock;
using Iceshrimp.Backend.Core.Database;
using Iceshrimp.Backend.Core.Database.Tables;
using Iceshrimp.Backend.Core.Helpers;
using Iceshrimp.Backend.Core.Middleware;
using Microsoft.EntityFrameworkCore;
namespace Iceshrimp.Backend.Core.Services;
public class SystemUserService(ILogger<SystemUserService> logger, DatabaseContext db)
{
private static readonly AsyncKeyedLocker<string> KeyedLocker = new(o =>
{
o.PoolSize = 10;
o.PoolInitialFill = 2;
});
public async Task<User> GetInstanceActorAsync()
{
return await GetOrCreateSystemUserAsync("instance.actor");
}
public async Task<User> GetRelayActorAsync()
{
return await GetOrCreateSystemUserAsync("relay.actor");
}
public async Task<(User user, UserKeypair keypair)> GetInstanceActorWithKeypairAsync()
{
return await GetOrCreateSystemUserAndKeypairAsync("instance.actor");
}
public async Task<(User user, UserKeypair keypair)> GetRelayActorWithKeypairAsync()
{
return await GetOrCreateSystemUserAndKeypairAsync("relay.actor");
}
private async Task<(User user, UserKeypair keypair)> GetOrCreateSystemUserAndKeypairAsync(string username)
{
var user = await GetOrCreateSystemUserAsync(username);
var keypair = await db.UserKeypairs.FirstOrDefaultAsync(p => p.User == user) ??
throw new Exception($"Failed to get keypair for system user {user.Id} ({username})");
return (user, keypair);
}
private async Task<User> GetOrCreateSystemUserAsync(string username)
{
var user = db.ChangeTracker
.Entries<User>()
.FirstOrDefault(p => p.Entity.UsernameLower == username.ToLowerInvariant() && p.Entity.IsLocalUser)
?.Entity;
user ??= await db.Users.FirstOrDefaultAsync(p => p.UsernameLower == username.ToLowerInvariant() &&
p.IsLocalUser);
if (user != null) return user;
using (await KeyedLocker.LockAsync(username.ToLowerInvariant()))
{
logger.LogTrace("GetOrCreateSystemUser delegate method called for user {username}", username);
return await CreateSystemUserAsync(username);
}
}
private async Task<User> CreateSystemUserAsync(string username)
{
if (await db.Users.AnyAsync(p => p.UsernameLower == username.ToLowerInvariant() && p.IsLocalUser))
throw new GracefulException($"System user {username} already exists");
var keypair = RSA.Create(4096);
var user = new User
{
Id = IdHelpers.GenerateSlowflakeId(),
CreatedAt = DateTime.UtcNow,
Username = username,
UsernameLower = username.ToLowerInvariant(),
Host = null,
IsAdmin = false,
IsLocked = true,
IsExplorable = false,
IsBot = true
};
var userKeypair = new UserKeypair
{
UserId = user.Id,
PrivateKey = keypair.ExportPkcs8PrivateKeyPem(),
PublicKey = keypair.ExportSubjectPublicKeyInfoPem()
};
var userProfile = new UserProfile { UserId = user.Id };
var userSettings = new UserSettings { UserId = user.Id, Password = null };
var usedUsername = new UsedUsername { CreatedAt = DateTime.UtcNow, Username = username.ToLowerInvariant() };
await db.AddRangeAsync(user, userKeypair, userProfile, userSettings, usedUsername);
await db.SaveChangesAsync();
return user;
}
}