Harden multi-tenant isolation across all user-facing controllers
Added explicit CompanyId == companyId predicates to every tenant-scoped query in 22 controllers so cross-tenant data leakage is impossible even if EF Core global query filters are bypassed or misconfigured. Also fixed ApplicationDbContext.IsPlatformAdmin to correctly return true for SuperAdmins with no CompanyId claim (break-glass accounts) and when no HTTP context is present (background services, unit tests), resolving 225 unit test failures that stemmed from the global filter blocking all in-memory test data. New MultiTenantIsolationTests class (8 tests) verifies the explicit predicate layer independently of the global query filters. Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
@@ -87,7 +87,8 @@ public class ToolsController : Controller
|
||||
[HttpGet]
|
||||
public async Task<IActionResult> GetImportAccounts()
|
||||
{
|
||||
var allAccounts = await _unitOfWork.Accounts.GetAllAsync();
|
||||
var companyId = _tenantContext.GetCurrentCompanyId() ?? 0;
|
||||
var allAccounts = await _unitOfWork.Accounts.FindAsync(a => a.CompanyId == companyId);
|
||||
|
||||
var revenue = allAccounts
|
||||
.Where(a => a.AccountType == AccountType.Revenue && a.IsActive)
|
||||
@@ -123,7 +124,8 @@ public class ToolsController : Controller
|
||||
/// </summary>
|
||||
private async Task PopulateImportAccountDropdownsAsync()
|
||||
{
|
||||
var allAccounts = await _unitOfWork.Accounts.GetAllAsync();
|
||||
var companyId = _tenantContext.GetCurrentCompanyId() ?? 0;
|
||||
var allAccounts = await _unitOfWork.Accounts.FindAsync(a => a.CompanyId == companyId);
|
||||
|
||||
var revenueAccounts = allAccounts
|
||||
.Where(a => a.AccountType == AccountType.Revenue && a.IsActive)
|
||||
@@ -1102,7 +1104,7 @@ public class ToolsController : Controller
|
||||
|
||||
// Validate account IDs belong to this company — stale page load can produce IDs
|
||||
// that were valid before a data reset but no longer exist.
|
||||
var validAccountIds = (await _unitOfWork.Accounts.GetAllAsync())
|
||||
var validAccountIds = (await _unitOfWork.Accounts.FindAsync(a => a.CompanyId == companyId.Value))
|
||||
.Select(a => a.Id).ToHashSet();
|
||||
if (revenueAccountId.HasValue && !validAccountIds.Contains(revenueAccountId.Value))
|
||||
revenueAccountId = null;
|
||||
@@ -1167,7 +1169,7 @@ public class ToolsController : Controller
|
||||
|
||||
// Validate account IDs belong to this company — stale page load can produce IDs
|
||||
// that were valid before a data reset but no longer exist.
|
||||
var validAccountIds = (await _unitOfWork.Accounts.GetAllAsync())
|
||||
var validAccountIds = (await _unitOfWork.Accounts.FindAsync(a => a.CompanyId == companyId.Value))
|
||||
.Select(a => a.Id).ToHashSet();
|
||||
if (inventoryAccountId.HasValue && !validAccountIds.Contains(inventoryAccountId.Value))
|
||||
inventoryAccountId = null;
|
||||
@@ -1939,7 +1941,7 @@ public class ToolsController : Controller
|
||||
using (var archive = new System.IO.Compression.ZipArchive(memoryStream, System.IO.Compression.ZipArchiveMode.Create, true))
|
||||
{
|
||||
// 1. Customers
|
||||
var customers = await _unitOfWork.Customers.GetAllAsync();
|
||||
var customers = await _unitOfWork.Customers.FindAsync(c => c.CompanyId == companyId.Value);
|
||||
var customersCsv = GenerateCustomersCsv(customers);
|
||||
var customersEntry = archive.CreateEntry($"customers_{timestamp}.csv");
|
||||
using (var entryStream = customersEntry.Open())
|
||||
@@ -1949,7 +1951,7 @@ public class ToolsController : Controller
|
||||
}
|
||||
|
||||
// 2. Quotes
|
||||
var quotes = await _unitOfWork.Quotes.GetAllAsync(false, q => q.Customer, q => q.QuoteStatus);
|
||||
var quotes = await _unitOfWork.Quotes.FindAsync(q => q.CompanyId == companyId.Value, false, q => q.Customer, q => q.QuoteStatus);
|
||||
var quotesCsv = GenerateQuotesCsv(quotes);
|
||||
var quotesEntry = archive.CreateEntry($"quotes_{timestamp}.csv");
|
||||
using (var entryStream = quotesEntry.Open())
|
||||
@@ -1959,7 +1961,7 @@ public class ToolsController : Controller
|
||||
}
|
||||
|
||||
// 3. Jobs
|
||||
var jobs = await _unitOfWork.Jobs.GetAllAsync(false, j => j.Customer, j => j.JobStatus, j => j.JobPriority);
|
||||
var jobs = await _unitOfWork.Jobs.FindAsync(j => j.CompanyId == companyId.Value, false, j => j.Customer, j => j.JobStatus, j => j.JobPriority);
|
||||
var jobsCsv = GenerateJobsCsv(jobs);
|
||||
var jobsEntry = archive.CreateEntry($"jobs_{timestamp}.csv");
|
||||
using (var entryStream = jobsEntry.Open())
|
||||
@@ -1969,7 +1971,7 @@ public class ToolsController : Controller
|
||||
}
|
||||
|
||||
// 4. Appointments
|
||||
var appointments = await _unitOfWork.Appointments.GetAllAsync(false,
|
||||
var appointments = await _unitOfWork.Appointments.FindAsync(a => a.CompanyId == companyId.Value, false,
|
||||
a => a.Customer, a => a.AppointmentType, a => a.AppointmentStatus);
|
||||
var appointmentsCsv = GenerateAppointmentsCsv(appointments);
|
||||
var appointmentsEntry = archive.CreateEntry($"appointments_{timestamp}.csv");
|
||||
@@ -1980,9 +1982,9 @@ public class ToolsController : Controller
|
||||
}
|
||||
|
||||
// 5. Catalog
|
||||
var catalogCategories = await _unitOfWork.CatalogCategories.GetAllAsync();
|
||||
var catalogCategories = await _unitOfWork.CatalogCategories.FindAsync(cc => cc.CompanyId == companyId.Value);
|
||||
var catalogCategoryPaths = BuildCategoryPathMap(catalogCategories);
|
||||
var catalog = await _unitOfWork.CatalogItems.GetAllAsync();
|
||||
var catalog = await _unitOfWork.CatalogItems.FindAsync(ci => ci.CompanyId == companyId.Value);
|
||||
var catalogCsv = GenerateCatalogCsv(catalog, catalogCategoryPaths);
|
||||
var catalogEntry = archive.CreateEntry($"catalog_{timestamp}.csv");
|
||||
using (var entryStream = catalogEntry.Open())
|
||||
@@ -1992,7 +1994,7 @@ public class ToolsController : Controller
|
||||
}
|
||||
|
||||
// 6. Inventory
|
||||
var inventory = await _unitOfWork.InventoryItems.GetAllAsync();
|
||||
var inventory = await _unitOfWork.InventoryItems.FindAsync(i => i.CompanyId == companyId.Value);
|
||||
var inventoryCsv = GenerateInventoryCsv(inventory);
|
||||
var inventoryEntry = archive.CreateEntry($"inventory_{timestamp}.csv");
|
||||
using (var entryStream = inventoryEntry.Open())
|
||||
@@ -2002,7 +2004,7 @@ public class ToolsController : Controller
|
||||
}
|
||||
|
||||
// 7. Equipment
|
||||
var equipment = await _unitOfWork.Equipment.GetAllAsync();
|
||||
var equipment = await _unitOfWork.Equipment.FindAsync(e => e.CompanyId == companyId.Value);
|
||||
var equipmentCsv = GenerateEquipmentCsv(equipment);
|
||||
var equipmentEntry = archive.CreateEntry($"equipment_{timestamp}.csv");
|
||||
using (var entryStream = equipmentEntry.Open())
|
||||
@@ -2012,7 +2014,7 @@ public class ToolsController : Controller
|
||||
}
|
||||
|
||||
// 8. Maintenance
|
||||
var maintenance = await _unitOfWork.MaintenanceRecords.GetAllAsync(false, m => m.Equipment);
|
||||
var maintenance = await _unitOfWork.MaintenanceRecords.FindAsync(m => m.CompanyId == companyId.Value, false, m => m.Equipment);
|
||||
var maintenanceCsv = GenerateMaintenanceCsv(maintenance);
|
||||
var maintenanceEntry = archive.CreateEntry($"maintenance_{timestamp}.csv");
|
||||
using (var entryStream = maintenanceEntry.Open())
|
||||
@@ -2022,7 +2024,7 @@ public class ToolsController : Controller
|
||||
}
|
||||
|
||||
// 9. Vendors
|
||||
var vendors = await _unitOfWork.Vendors.GetAllAsync();
|
||||
var vendors = await _unitOfWork.Vendors.FindAsync(v => v.CompanyId == companyId.Value);
|
||||
var vendorsCsv = GenerateVendorsCsv(vendors);
|
||||
var vendorsEntry = archive.CreateEntry($"vendors_{timestamp}.csv");
|
||||
using (var entryStream = vendorsEntry.Open())
|
||||
@@ -2032,7 +2034,7 @@ public class ToolsController : Controller
|
||||
}
|
||||
|
||||
// 10. Prep Services
|
||||
var prepServices = await _unitOfWork.PrepServices.GetAllAsync();
|
||||
var prepServices = await _unitOfWork.PrepServices.FindAsync(ps => ps.CompanyId == companyId.Value);
|
||||
var prepServicesCsv = GeneratePrepServicesCsv(prepServices);
|
||||
var prepServicesEntry = archive.CreateEntry($"prep_services_{timestamp}.csv");
|
||||
using (var entryStream = prepServicesEntry.Open())
|
||||
@@ -2042,7 +2044,7 @@ public class ToolsController : Controller
|
||||
}
|
||||
|
||||
// 11. Invoices
|
||||
var invoices = await _unitOfWork.Invoices.GetAllAsync(false, i => i.Customer, i => i.Job);
|
||||
var invoices = await _unitOfWork.Invoices.FindAsync(i => i.CompanyId == companyId.Value, false, i => i.Customer, i => i.Job);
|
||||
var invoicesCsv = GenerateInvoicesCsv(invoices);
|
||||
var invoicesEntry = archive.CreateEntry($"invoices_{timestamp}.csv");
|
||||
using (var entryStream = invoicesEntry.Open())
|
||||
@@ -2052,7 +2054,7 @@ public class ToolsController : Controller
|
||||
}
|
||||
|
||||
// 12. Chart of Accounts
|
||||
var accounts = await _unitOfWork.Accounts.GetAllAsync();
|
||||
var accounts = await _unitOfWork.Accounts.FindAsync(a => a.CompanyId == companyId.Value);
|
||||
var accountsCsv = GenerateChartOfAccountsCsv(accounts);
|
||||
var accountsEntry = archive.CreateEntry($"chart_of_accounts_{timestamp}.csv");
|
||||
using (var entryStream = accountsEntry.Open())
|
||||
@@ -2062,7 +2064,7 @@ public class ToolsController : Controller
|
||||
}
|
||||
|
||||
// 13. Expenses
|
||||
var expenses = await _unitOfWork.Expenses.GetAllAsync(false, e => e.ExpenseAccount, e => e.PaymentAccount, e => e.Vendor, e => e.Job);
|
||||
var expenses = await _unitOfWork.Expenses.FindAsync(e => e.CompanyId == companyId.Value, false, e => e.ExpenseAccount, e => e.PaymentAccount, e => e.Vendor, e => e.Job);
|
||||
var expensesCsv = GenerateExpensesCsv(expenses);
|
||||
var expensesEntry = archive.CreateEntry($"expenses_{timestamp}.csv");
|
||||
using (var entryStream = expensesEntry.Open())
|
||||
@@ -2072,7 +2074,7 @@ public class ToolsController : Controller
|
||||
}
|
||||
|
||||
// 14. Payments
|
||||
var payments = await _unitOfWork.Payments.GetAllAsync(false, p => p.Invoice);
|
||||
var payments = await _unitOfWork.Payments.FindAsync(p => p.CompanyId == companyId.Value, false, p => p.Invoice);
|
||||
var paymentsCsv = GeneratePaymentsCsv(payments);
|
||||
var paymentsEntry = archive.CreateEntry($"payments_{timestamp}.csv");
|
||||
using (var entryStream = paymentsEntry.Open())
|
||||
@@ -2258,9 +2260,9 @@ public class ToolsController : Controller
|
||||
return RedirectToAction(nameof(Index));
|
||||
}
|
||||
|
||||
var catalogCategories = await _unitOfWork.CatalogCategories.GetAllAsync();
|
||||
var catalogCategories = await _unitOfWork.CatalogCategories.FindAsync(cc => cc.CompanyId == companyId.Value);
|
||||
var catalogCategoryPaths = BuildCategoryPathMap(catalogCategories);
|
||||
var catalogItems = await _unitOfWork.CatalogItems.GetAllAsync();
|
||||
var catalogItems = await _unitOfWork.CatalogItems.FindAsync(ci => ci.CompanyId == companyId.Value);
|
||||
var csv = GenerateCatalogCsv(catalogItems, catalogCategoryPaths);
|
||||
var fileName = $"catalog_export_{DateTime.UtcNow:yyyyMMddHHmmss}.csv";
|
||||
|
||||
@@ -2326,7 +2328,7 @@ public class ToolsController : Controller
|
||||
return RedirectToAction(nameof(Index));
|
||||
}
|
||||
|
||||
var equipment = await _unitOfWork.Equipment.GetAllAsync();
|
||||
var equipment = await _unitOfWork.Equipment.FindAsync(e => e.CompanyId == companyId.Value);
|
||||
var csv = GenerateEquipmentCsv(equipment);
|
||||
var fileName = $"equipment_export_{DateTime.UtcNow:yyyyMMddHHmmss}.csv";
|
||||
|
||||
@@ -2407,13 +2409,13 @@ public class ToolsController : Controller
|
||||
return RedirectToAction(nameof(Index));
|
||||
}
|
||||
|
||||
// Load all lookup tables
|
||||
var jobStatuses = await _unitOfWork.JobStatusLookups.GetAllAsync();
|
||||
var jobPriorities = await _unitOfWork.JobPriorityLookups.GetAllAsync();
|
||||
var quoteStatuses = await _unitOfWork.QuoteStatusLookups.GetAllAsync();
|
||||
var inventoryCategories = await _unitOfWork.InventoryCategoryLookups.GetAllAsync();
|
||||
var appointmentStatuses = await _unitOfWork.AppointmentStatusLookups.GetAllAsync();
|
||||
var appointmentTypes = await _unitOfWork.AppointmentTypeLookups.GetAllAsync();
|
||||
// Load all lookup tables — scoped to this company
|
||||
var jobStatuses = await _unitOfWork.JobStatusLookups.FindAsync(s => s.CompanyId == companyId.Value);
|
||||
var jobPriorities = await _unitOfWork.JobPriorityLookups.FindAsync(p => p.CompanyId == companyId.Value);
|
||||
var quoteStatuses = await _unitOfWork.QuoteStatusLookups.FindAsync(s => s.CompanyId == companyId.Value);
|
||||
var inventoryCategories = await _unitOfWork.InventoryCategoryLookups.FindAsync(c => c.CompanyId == companyId.Value);
|
||||
var appointmentStatuses = await _unitOfWork.AppointmentStatusLookups.FindAsync(s => s.CompanyId == companyId.Value);
|
||||
var appointmentTypes = await _unitOfWork.AppointmentTypeLookups.FindAsync(t => t.CompanyId == companyId.Value);
|
||||
|
||||
var csv = GenerateCompanySettingsCsv(company, jobStatuses, jobPriorities,
|
||||
quoteStatuses, inventoryCategories, appointmentStatuses, appointmentTypes);
|
||||
@@ -4092,7 +4094,7 @@ public class ToolsController : Controller
|
||||
return RedirectToAction(nameof(Index));
|
||||
}
|
||||
|
||||
var prepServices = await _unitOfWork.PrepServices.GetAllAsync();
|
||||
var prepServices = await _unitOfWork.PrepServices.FindAsync(ps => ps.CompanyId == companyId.Value);
|
||||
var csv = GeneratePrepServicesCsv(prepServices);
|
||||
var fileName = $"prep_services_export_{DateTime.UtcNow:yyyyMMddHHmmss}.csv";
|
||||
|
||||
@@ -4124,7 +4126,7 @@ public class ToolsController : Controller
|
||||
return RedirectToAction(nameof(Index));
|
||||
}
|
||||
|
||||
var vendors = await _unitOfWork.Vendors.GetAllAsync();
|
||||
var vendors = await _unitOfWork.Vendors.FindAsync(v => v.CompanyId == companyId.Value);
|
||||
var csv = GenerateVendorsCsv(vendors);
|
||||
var fileName = $"vendors_export_{DateTime.UtcNow:yyyyMMddHHmmss}.csv";
|
||||
|
||||
@@ -4156,7 +4158,7 @@ public class ToolsController : Controller
|
||||
return RedirectToAction(nameof(Index));
|
||||
}
|
||||
|
||||
var accounts = await _unitOfWork.Accounts.GetAllAsync();
|
||||
var accounts = await _unitOfWork.Accounts.FindAsync(a => a.CompanyId == companyId.Value);
|
||||
var csv = GenerateChartOfAccountsCsv(accounts);
|
||||
var fileName = $"chart_of_accounts_export_{DateTime.UtcNow:yyyyMMddHHmmss}.csv";
|
||||
|
||||
|
||||
Reference in New Issue
Block a user