Files
Moonlight/Moonlight/App/Services/Store/StoreOrderService.cs
2023-10-23 13:27:49 +02:00

198 lines
6.9 KiB
C#

using Microsoft.EntityFrameworkCore;
using Moonlight.App.Database.Entities;
using Moonlight.App.Database.Entities.Store;
using Moonlight.App.Exceptions;
using Moonlight.App.Repositories;
using Moonlight.App.Services.ServiceManage;
namespace Moonlight.App.Services.Store;
public class StoreOrderService
{
private readonly IServiceScopeFactory ServiceScopeFactory;
public StoreOrderService(IServiceScopeFactory serviceScopeFactory)
{
ServiceScopeFactory = serviceScopeFactory;
}
public Task Validate(User u, Product p, int durationMultiplier, Coupon? c)
{
using var scope = ServiceScopeFactory.CreateScope();
var productRepo = scope.ServiceProvider.GetRequiredService<Repository<Product>>();
var userRepo = scope.ServiceProvider.GetRequiredService<Repository<User>>();
var serviceRepo = scope.ServiceProvider.GetRequiredService<Repository<Service>>();
var couponRepo = scope.ServiceProvider.GetRequiredService<Repository<Coupon>>();
// Ensure the values are safe and loaded by using the created scope to bypass the cache
var user = userRepo
.Get()
.Include(x => x.CouponUses)
.ThenInclude(x => x.Coupon)
.FirstOrDefault(x => x.Id == u.Id);
if (user == null)
throw new DisplayException("Unsafe value detected. Please reload the page to proceed");
var product = productRepo
.Get()
.FirstOrDefault(x => x.Id == p.Id);
if (product == null)
throw new DisplayException("Unsafe value detected. Please reload the page to proceed");
Coupon? coupon = c;
if (coupon != null) // Only check if the coupon actually has a value
{
coupon = couponRepo
.Get()
.FirstOrDefault(x => x.Id == coupon.Id);
if (coupon == null)
throw new DisplayException("Unsafe value detected. Please reload the page to proceed");
}
// Perform checks on selected order
if (coupon != null && user.CouponUses.Any(x => x.Coupon.Id == coupon.Id))
throw new DisplayException("Coupon already used");
if (coupon != null && coupon.Amount < 1)
throw new DisplayException("No coupon uses left");
var price = product.Price * durationMultiplier;
if (coupon != null)
price = Math.Round(price - (price * coupon.Percent / 100), 2);
if (user.Balance < price)
throw new DisplayException("Order is too expensive");
var userServices = serviceRepo
.Get()
.Where(x => x.Product.Id == product.Id)
.Count(x => x.Owner.Id == user.Id);
if (userServices >= product.MaxPerUser)
throw new DisplayException("The limit of this product on your account has been reached");
if (product.Stock < 1)
throw new DisplayException("The product is out of stock");
return Task.CompletedTask;
}
public async Task<Service> Process(User u, Product p, int durationMultiplier, Coupon? c)
{
// Validate to ensure we dont process an illegal order
await Validate(u, p, durationMultiplier, c);
// Create scope and get required services
using var scope = ServiceScopeFactory.CreateScope();
var serviceService = scope.ServiceProvider.GetRequiredService<ServiceService>();
var transactionService = scope.ServiceProvider.GetRequiredService<TransactionService>();
// Calculate price
var price = p.Price * durationMultiplier;
if (c != null)
price = Math.Round(price - (price * c.Percent / 100), 2);
// Calculate duration
var duration = durationMultiplier * p.Duration;
// Add transaction
await transactionService.Add(u, -1 * price, $"Bought product '{p.Name}' for {duration} days");
// Process coupon if used
if (c != null)
{
// Remove one use from the coupon
var couponRepo = scope.ServiceProvider.GetRequiredService<Repository<Coupon>>();
var coupon = couponRepo
.Get()
.First(x => x.Id == c.Id);
coupon.Amount--;
couponRepo.Update(coupon);
// Add coupon use to user
var userRepo = scope.ServiceProvider.GetRequiredService<Repository<User>>();
var user = userRepo
.Get()
.Include(x => x.CouponUses)
.First(x => x.Id == u.Id);
user.CouponUses.Add(new ()
{
Coupon = coupon
});
userRepo.Update(user);
}
// Create service
return await serviceService.Admin.Create(u, p,
service => { service.RenewAt = DateTime.UtcNow.AddDays(duration); });
}
public Task ValidateRenew(User u, Service s, int durationMultiplier)
{
using var scope = ServiceScopeFactory.CreateScope();
var userRepo = scope.ServiceProvider.GetRequiredService<Repository<User>>();
var serviceRepo = scope.ServiceProvider.GetRequiredService<Repository<Service>>();
var user = userRepo.Get().FirstOrDefault(x => x.Id == u.Id);
if(user == null)
throw new DisplayException("Unsafe value detected. Please reload the page to proceed");
var service = serviceRepo
.Get()
.Include(x => x.Product)
.Include(x => x.Owner)
.FirstOrDefault(x => x.Id == s.Id);
if(service == null)
throw new DisplayException("Unsafe value detected. Please reload the page to proceed");
var price = service.Product.Price * durationMultiplier;
if (user.Balance < price)
throw new DisplayException("Order is too expensive");
return Task.CompletedTask;
}
public async Task Renew(User u, Service s, int durationMultiplier)
{
await ValidateRenew(u, s, durationMultiplier);
using var scope = ServiceScopeFactory.CreateScope();
var serviceRepo = scope.ServiceProvider.GetRequiredService<Repository<Service>>();
var transactionService = scope.ServiceProvider.GetRequiredService<TransactionService>();
var service = serviceRepo
.Get()
.Include(x => x.Product)
.Include(x => x.Owner)
.First(x => x.Id == s.Id);
var price = service.Product.Price * durationMultiplier;
// Calculate duration
var duration = durationMultiplier * service.Product.Duration;
// Add transaction
await transactionService.Add(u, -1 * price, $"Renewed service '{service.Nickname ?? $"Service {service.Id}"}' for {duration} days");
service.RenewAt = service.RenewAt.AddDays(duration);
serviceRepo.Update(service);
}
}