using ZB.MOM.WW.ScadaBridge.AuditLog; using ZB.MOM.WW.ScadaBridge.ClusterInfrastructure; using ZB.MOM.WW.ScadaBridge.Communication; using ZB.MOM.WW.ScadaBridge.Commons.Interfaces.Services; using ZB.MOM.WW.ScadaBridge.DataConnectionLayer; using ZB.MOM.WW.ScadaBridge.ExternalSystemGateway; using ZB.MOM.WW.ScadaBridge.HealthMonitoring; using ZB.MOM.WW.ScadaBridge.Host.Actors; using ZB.MOM.WW.ScadaBridge.Host.Health; using ZB.MOM.WW.ScadaBridge.NotificationService; using ZB.MOM.WW.ScadaBridge.SiteEventLogging; using ZB.MOM.WW.ScadaBridge.SiteRuntime; using ZB.MOM.WW.ScadaBridge.StoreAndForward; namespace ZB.MOM.WW.ScadaBridge.Host; /// /// Extracted site-role DI registrations so both Program.cs and tests /// use the same composition root. /// public static class SiteServiceRegistration { /// Registers all DI services required for the site role. /// The service collection to register into. /// Application configuration for options binding. public static void Configure(IServiceCollection services, IConfiguration config) { // Shared components services.AddClusterInfrastructure(); services.AddCommunication(); services.AddSiteHealthMonitoring(); services.AddExternalSystemGateway(); // AddNotificationService() is intentionally NOT registered on the site path. // Sites no longer deliver notifications over SMTP — a buffered notification is // forwarded to the central cluster (via NotificationForwarder / SiteCommunicationActor), // and central owns SMTP delivery through the Notification Outbox. The SMTP machinery // (OAuth2TokenService, ISmtpClientWrapper) has no consumer on a site node. // Health report transport: sends SiteHealthReport to SiteCommunicationActor via Akka services.AddSingleton(); services.AddSingleton(); // Site-only components — AddSiteRuntime registers SiteStorageService with SQLite path // and site-local repository implementations (IExternalSystemRepository, INotificationRepository) var siteDbPath = config["ScadaBridge:Database:SiteDbPath"] ?? "site.db"; services.AddSiteRuntime($"Data Source={siteDbPath}"); services.AddDataConnectionLayer(); // Audit Log #23 (M3 Bundle F): adapter that surfaces the site id to // StoreAndForwardService through DI WITHOUT introducing a // StoreAndForward → HealthMonitoring project-reference cycle. Must be // registered BEFORE AddStoreAndForward so the S&F factory resolves a // non-empty SiteId at construction time (otherwise the S&F service is // a singleton and the empty-string value would be cached for the // lifetime of the process). services.AddSingleton(); services.AddStoreAndForward(); services.AddSiteEventLogging(); // Audit Log (#23) — site-side hot-path writer + telemetry collaborators. // The SiteAuditTelemetryActor itself is registered by AkkaHostedService // in the site-role block; this call wires every DI dependency it (and // ScriptRuntimeContext, when Bundle F lands) reaches for. services.AddAuditLog(config); // Audit Log (#23) M2 Bundle G — bridge FallbackAuditWriter primary // failures into the site health report payload as // SiteAuditWriteFailures. Must come AFTER both AddSiteHealthMonitoring // (registers ISiteHealthCollector) and AddAuditLog (registers the // NoOp default this call replaces). services.AddAuditLogHealthMetricsBridge(); // WP-13: Akka.NET bootstrap via hosted service services.AddSingleton(); services.AddHostedService(sp => sp.GetRequiredService()); // The shared ZB.MOM.WW.Health Akka checks resolve ActorSystem from DI. ScadaBridge owns the // ActorSystem inside AkkaHostedService (not a DI singleton), so bridge it as TRANSIENT: each // resolve re-reads the current value — null while warming up (checks → Degraded), live after. // The factory must NOT throw: GetService() must return null (not raise) pre-start. services.AddTransient(sp => sp.GetRequiredService().ActorSystem!); // Cluster node status provider for health reports services.AddSingleton(sp => { var akkaService = sp.GetRequiredService(); var nodeOptions = sp.GetRequiredService>().Value; var siteRole = $"site-{nodeOptions.SiteId}"; return new AkkaClusterNodeProvider(akkaService, siteRole); }); // Options binding BindSharedOptions(services, config); services.Configure(config.GetSection("ScadaBridge:SiteRuntime")); services.Configure(config.GetSection("ScadaBridge:DataConnection")); services.Configure(config.GetSection("ScadaBridge:StoreAndForward")); services.Configure(config.GetSection("ScadaBridge:SiteEventLog")); } /// Binds shared options sections (Node, Cluster, Database, Communication, etc.) used by both site and central roles. /// The service collection to bind options into. /// Application configuration supplying the option values. public static void BindSharedOptions(IServiceCollection services, IConfiguration config) { services.Configure(config.GetSection("ScadaBridge:Node")); services.Configure(config.GetSection("ScadaBridge:Cluster")); services.Configure(config.GetSection("ScadaBridge:Database")); services.Configure(config.GetSection("ScadaBridge:Communication")); services.Configure(config.GetSection("ScadaBridge:HealthMonitoring")); services.Configure(config.GetSection("ScadaBridge:Notification")); services.Configure(config.GetSection("ScadaBridge:Logging")); // Audit Log (#23) — exposes ScadaBridge:Node:NodeName to downstream audit // writers so they can stamp the SourceNode column. Registered here in // shared bootstrap because every node (central + site) needs it. services.AddSingleton(); } }