using Azaion.Common; using Azaion.Common.Database; using Azaion.Common.Entities; using Azaion.Common.Requests; using LinqToDB; namespace Azaion.Services; public interface IUserService { Task RegisterUser(RegisterUserRequest request, CancellationToken cancellationToken = default); Task ValidateUser(LoginRequest request, string? hardwareId = null, CancellationToken cancellationToken = default); Task UpdateHardwareId(string email, string hardwareId, CancellationToken cancellationToken = default); } public class UserService(IDbFactory dbFactory) : IUserService { public async Task RegisterUser(RegisterUserRequest request, CancellationToken cancellationToken = default) { await dbFactory.RunAdmin(async db => { var existingUser = await db.Users.FirstOrDefaultAsync(u => u.Email == request.Email, token: cancellationToken); if (existingUser != null) throw new BusinessException(ExceptionEnum.UserExists, "User already exists"); await db.InsertAsync(new User { Id = Guid.NewGuid(), Email = request.Email, PasswordHash = request.Password.ToHash(), Role = request.Role }, token: cancellationToken); }); } public async Task ValidateUser(LoginRequest request, string? hardwareId = null, CancellationToken cancellationToken = default) => await dbFactory.Run(async db => { var user = await db.Users.FirstOrDefaultAsync(x => x.Email == request.Email, token: cancellationToken); if (user == null) throw new BusinessException(ExceptionEnum.NoUserFound, "No user found"); if (request.Password.ToHash() != user.PasswordHash) throw new BusinessException(ExceptionEnum.PasswordIncorrect, "Passwords do not match"); if (user.Role == RoleEnum.ApiAdmin) return user; // For Non-API admins hardwareId should match if it was already set if (user.HardwareId != null && user.HardwareId != hardwareId) throw new BusinessException(ExceptionEnum.HardwareIdMismatch, "Hardware id mismatch"); return user; }); public async Task UpdateHardwareId(string email, string hardwareId, CancellationToken cancellationToken = default) => await dbFactory.RunAdmin(async db => await db.Users.UpdateAsync(x => x.Email == email, u => new User { HardwareId = hardwareId}, token: cancellationToken)); }