Phases 3 & 4: Complete data access architecture migration
Phase 3 — eliminated ApplicationDbContext from all non-exempt controllers, routing all data access through IUnitOfWork. Added IPlainRepository<T> for the four platform entities (Announcement, BannedIp, DashboardTip, ReleaseNote) that intentionally don't extend BaseEntity and therefore can't use the constrained IRepository<T>. Added permanent-exception comments to the 18 controllers that legitimately retain direct DbContext access (Identity infra, cross-tenant platform ops, bulk streaming exports). Phase 4 — added EnforceDataAccessArchitecture() to Program.cs, a startup gate that reflects over every Controller subclass and throws at boot if any non-exempt controller injects ApplicationDbContext. The app cannot start with a violation. Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
@@ -9,7 +9,6 @@ using PowderCoating.Application.DTOs.User;
|
||||
using PowderCoating.Application.Interfaces;
|
||||
using PowderCoating.Core.Entities;
|
||||
using PowderCoating.Core.Interfaces;
|
||||
using PowderCoating.Infrastructure.Data;
|
||||
using PowderCoating.Shared.Constants;
|
||||
|
||||
namespace PowderCoating.Web.Controllers;
|
||||
@@ -25,7 +24,6 @@ public class CompanyUsersController : Controller
|
||||
private readonly ILogger<CompanyUsersController> _logger;
|
||||
private readonly IUnitOfWork _unitOfWork;
|
||||
private readonly ISubscriptionService _subscriptionService;
|
||||
private readonly ApplicationDbContext _context;
|
||||
private readonly IEmailService _emailService;
|
||||
|
||||
public CompanyUsersController(
|
||||
@@ -34,7 +32,6 @@ public class CompanyUsersController : Controller
|
||||
ILogger<CompanyUsersController> logger,
|
||||
IUnitOfWork unitOfWork,
|
||||
ISubscriptionService subscriptionService,
|
||||
ApplicationDbContext context,
|
||||
IEmailService emailService)
|
||||
{
|
||||
_userManager = userManager;
|
||||
@@ -42,7 +39,6 @@ public class CompanyUsersController : Controller
|
||||
_logger = logger;
|
||||
_unitOfWork = unitOfWork;
|
||||
_subscriptionService = subscriptionService;
|
||||
_context = context;
|
||||
_emailService = emailService;
|
||||
}
|
||||
|
||||
@@ -372,8 +368,8 @@ public class CompanyUsersController : Controller
|
||||
CompanyId = companyId!.Value
|
||||
};
|
||||
|
||||
await _context.ShopWorkers.AddAsync(shopWorker);
|
||||
await _context.SaveChangesAsync();
|
||||
await _unitOfWork.ShopWorkers.AddAsync(shopWorker);
|
||||
await _unitOfWork.CompleteAsync();
|
||||
|
||||
_logger.LogInformation("ShopWorker record created for user {Email}", user.Email);
|
||||
}
|
||||
@@ -639,9 +635,8 @@ public class CompanyUsersController : Controller
|
||||
{
|
||||
// Search by oldEmail so we find the record even when the email just changed
|
||||
var lookupEmail = emailChanged ? oldEmail : user.Email;
|
||||
var existingShopWorker = await _context.ShopWorkers
|
||||
.Where(sw => sw.Email == lookupEmail && sw.CompanyId == user.CompanyId)
|
||||
.ToListAsync();
|
||||
var existingShopWorker = (await _unitOfWork.ShopWorkers.FindAsync(
|
||||
sw => sw.Email == lookupEmail && sw.CompanyId == user.CompanyId)).ToList();
|
||||
|
||||
if (!existingShopWorker.Any())
|
||||
{
|
||||
@@ -656,8 +651,8 @@ public class CompanyUsersController : Controller
|
||||
CompanyId = user.CompanyId
|
||||
};
|
||||
|
||||
await _context.ShopWorkers.AddAsync(shopWorker);
|
||||
await _context.SaveChangesAsync();
|
||||
await _unitOfWork.ShopWorkers.AddAsync(shopWorker);
|
||||
await _unitOfWork.CompleteAsync();
|
||||
|
||||
_logger.LogInformation("ShopWorker record created for user {Email}", user.Email);
|
||||
}
|
||||
@@ -684,7 +679,7 @@ public class CompanyUsersController : Controller
|
||||
shopWorker.Phone = user.PhoneNumber;
|
||||
|
||||
if (shopWorkerDirty)
|
||||
await _context.SaveChangesAsync();
|
||||
await _unitOfWork.CompleteAsync();
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
Reference in New Issue
Block a user