using System.Security.Claims; using Bunit; using Microsoft.AspNetCore.Components; using Microsoft.AspNetCore.Components.Authorization; using Microsoft.Extensions.DependencyInjection; using NSubstitute; using ScadaLink.CentralUI.Auth; using ScadaLink.Commons.Entities.Instances; using ScadaLink.Commons.Entities.Sites; using ScadaLink.Commons.Entities.Templates; using ScadaLink.Commons.Interfaces.Repositories; using ScadaLink.Commons.Interfaces.Services; using ScadaLink.DeploymentManager; using ScadaLink.Security; using ScadaLink.TemplateEngine.Services; using InstanceConfigurePage = ScadaLink.CentralUI.Components.Pages.Deployment.InstanceConfigure; namespace ScadaLink.CentralUI.Tests.Deployment; /// /// Bundle D drill-in test (#23 M7-T12) for the Instance Configure page. The /// chip routes operators into the central Audit Log pre-filtered by /// ?instance={Instance.UniqueName}. Instance is UI-only on the filter /// bar (the repository filter contract has no instance column), so the page /// uses the UI-text seam — the Audit Log's filter bar pre-populates its /// Instance free-text input from this query string. /// public class InstanceConfigureAuditDrillinTests : BunitContext { private readonly ITemplateEngineRepository _templateRepo = Substitute.For(); private readonly ISiteRepository _siteRepo = Substitute.For(); public InstanceConfigureAuditDrillinTests() { // Loose JS interop because shared components on the page render // localStorage / clipboard touches that we don't care about here. JSInterop.Mode = JSRuntimeMode.Loose; Services.AddSingleton(_templateRepo); Services.AddSingleton(_siteRepo); Services.AddSingleton(new InstanceService(_templateRepo, Substitute.For())); Services.AddSingleton(Substitute.For()); // Auth: a system-wide Deployment user so SiteScope grants everything. var claims = new[] { new Claim("Username", "deployer"), new Claim(JwtTokenService.RoleClaimType, "Deployment"), }; var user = new ClaimsPrincipal(new ClaimsIdentity(claims, "TestAuth")); var authProvider = new TestAuthStateProvider(user); Services.AddSingleton(authProvider); Services.AddSingleton(new SiteScopeService(authProvider)); Services.AddAuthorizationCore(); AuthorizationPolicies.AddScadaLinkAuthorization(Services); } [Fact] public void Page_HasRecentAuditActivityLink_WithInstanceUniqueName() { var instance = new Instance("Pump-Station-007") { Id = 42, TemplateId = 1, SiteId = 1, State = ScadaLink.Commons.Types.Enums.InstanceState.NotDeployed, }; _templateRepo.GetInstanceByIdAsync(42, Arg.Any()).Returns(instance); _templateRepo.GetTemplateByIdAsync(1, Arg.Any()) .Returns(new Template("Pump") { Id = 1 }); _siteRepo.GetAllSitesAsync(Arg.Any()) .Returns(new List { new("Plant A", "plant-a") { Id = 1 } }); _templateRepo.GetAreasBySiteIdAsync(1, Arg.Any()) .Returns(new List()); _templateRepo.GetAttributesByTemplateIdAsync(1, Arg.Any()) .Returns(new List()); _siteRepo.GetDataConnectionsBySiteIdAsync(1, Arg.Any()) .Returns(new List()); _templateRepo.GetBindingsByInstanceIdAsync(42, Arg.Any()) .Returns(new List()); _templateRepo.GetOverridesByInstanceIdAsync(42, Arg.Any()) .Returns(new List()); _templateRepo.GetAlarmsByTemplateIdAsync(1, Arg.Any()) .Returns(new List()); _templateRepo.GetAlarmOverridesByInstanceIdAsync(42, Arg.Any()) .Returns(new List()); var cut = Render(p => p.Add(c => c.Id, 42)); cut.WaitForAssertion(() => { var link = cut.Find("a[data-test=\"audit-link\"]"); Assert.Equal("/audit/log?instance=Pump-Station-007", link.GetAttribute("href")); Assert.Contains("Recent audit activity", link.TextContent); }); } }