feat(host): role-gated Program.cs composes all v2 components
This commit is contained in:
@@ -0,0 +1,24 @@
|
|||||||
|
using Microsoft.EntityFrameworkCore;
|
||||||
|
using Microsoft.Extensions.Configuration;
|
||||||
|
using Microsoft.Extensions.DependencyInjection;
|
||||||
|
|
||||||
|
namespace ZB.MOM.WW.OtOpcUa.Configuration;
|
||||||
|
|
||||||
|
public static class ServiceCollectionExtensions
|
||||||
|
{
|
||||||
|
public const string ConnectionStringName = "ConfigDb";
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Registers <see cref="IDbContextFactory{TContext}"/> for <see cref="OtOpcUaConfigDbContext"/>
|
||||||
|
/// using the connection string named <c>ConfigDb</c> from <see cref="IConfiguration"/>.
|
||||||
|
/// </summary>
|
||||||
|
public static IServiceCollection AddOtOpcUaConfigDb(this IServiceCollection services, IConfiguration configuration)
|
||||||
|
{
|
||||||
|
var connectionString = configuration.GetConnectionString(ConnectionStringName)
|
||||||
|
?? throw new InvalidOperationException(
|
||||||
|
$"Connection string '{ConnectionStringName}' is required. Add it to appsettings.json or the OTOPCUA_CONFIG_CONNECTION env var.");
|
||||||
|
|
||||||
|
services.AddDbContextFactory<OtOpcUaConfigDbContext>(opt => opt.UseSqlServer(connectionString));
|
||||||
|
return services;
|
||||||
|
}
|
||||||
|
}
|
||||||
19
src/Server/ZB.MOM.WW.OtOpcUa.Host/App.razor
Normal file
19
src/Server/ZB.MOM.WW.OtOpcUa.Host/App.razor
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
@* Root Blazor component for the fused OtOpcUa.Host. Pulls in the AdminUI library's
|
||||||
|
_Imports + the Deployments page. The full layout (sidebar, top bar, etc.) is part of
|
||||||
|
the legacy Admin migration tracked as F15 — for now this is the bare minimum that lets
|
||||||
|
the Razor pipeline render. *@
|
||||||
|
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="en">
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8" />
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
||||||
|
<base href="/" />
|
||||||
|
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bootstrap@5.3.2/dist/css/bootstrap.min.css" />
|
||||||
|
<HeadOutlet @rendermode="InteractiveServer" />
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<Routes @rendermode="InteractiveServer" />
|
||||||
|
<script src="_framework/blazor.web.js"></script>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
11
src/Server/ZB.MOM.WW.OtOpcUa.Host/MainLayout.razor
Normal file
11
src/Server/ZB.MOM.WW.OtOpcUa.Host/MainLayout.razor
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
@inherits LayoutComponentBase
|
||||||
|
|
||||||
|
<div class="container mt-3">
|
||||||
|
<nav class="d-flex gap-3 mb-3 border-bottom pb-2">
|
||||||
|
<strong>OtOpcUa</strong>
|
||||||
|
<a href="/deployments">Deployments</a>
|
||||||
|
</nav>
|
||||||
|
<main>
|
||||||
|
@Body
|
||||||
|
</main>
|
||||||
|
</div>
|
||||||
@@ -1,4 +1,87 @@
|
|||||||
|
using Akka.Hosting;
|
||||||
|
using Serilog;
|
||||||
|
using ZB.MOM.WW.OtOpcUa.AdminUI;
|
||||||
|
using ZB.MOM.WW.OtOpcUa.AdminUI.Clients;
|
||||||
|
using ZB.MOM.WW.OtOpcUa.AdminUI.Hubs;
|
||||||
|
using ZB.MOM.WW.OtOpcUa.Cluster;
|
||||||
|
using ZB.MOM.WW.OtOpcUa.Configuration;
|
||||||
|
using ZB.MOM.WW.OtOpcUa.ControlPlane;
|
||||||
|
using ZB.MOM.WW.OtOpcUa.Host;
|
||||||
|
using ZB.MOM.WW.OtOpcUa.Host.Health;
|
||||||
|
using ZB.MOM.WW.OtOpcUa.Security;
|
||||||
|
using ZB.MOM.WW.OtOpcUa.Security.Endpoints;
|
||||||
|
|
||||||
|
// Roles drive the entire conditional wiring below — see ZB.MOM.WW.OtOpcUa.Cluster.RoleParser.
|
||||||
|
var roles = RoleParser.Parse(Environment.GetEnvironmentVariable("OTOPCUA_ROLES"));
|
||||||
|
var hasAdmin = roles.Contains("admin");
|
||||||
|
var hasDriver = roles.Contains("driver");
|
||||||
|
|
||||||
var builder = WebApplication.CreateBuilder(args);
|
var builder = WebApplication.CreateBuilder(args);
|
||||||
|
|
||||||
|
// Per-role appsettings overlay: appsettings.{role}.json (single role) or appsettings.admin-driver.json
|
||||||
|
// (both). Optional — base appsettings.json carries enough to boot if these don't exist.
|
||||||
|
var roleSuffix = roles.Length == 0 ? null : string.Join('-', roles.OrderBy(r => r, StringComparer.Ordinal));
|
||||||
|
if (roleSuffix is not null)
|
||||||
|
builder.Configuration.AddJsonFile($"appsettings.{roleSuffix}.json", optional: true, reloadOnChange: true);
|
||||||
|
|
||||||
|
// Serilog — rolling daily file sink per CLAUDE.md. Console for local dev.
|
||||||
|
builder.Host.UseSerilog((ctx, lc) => lc
|
||||||
|
.ReadFrom.Configuration(ctx.Configuration)
|
||||||
|
.WriteTo.Console()
|
||||||
|
.WriteTo.File("logs/otopcua-.log", rollingInterval: RollingInterval.Day));
|
||||||
|
|
||||||
|
// Windows-service registration is handled at install time by scripts/install/Install-Services.ps1
|
||||||
|
// (Task 62) rather than in-process, so the binary stays cross-platform-compilable.
|
||||||
|
|
||||||
|
// Shared services — always registered regardless of role. ConfigDb is required for everything.
|
||||||
|
builder.Services.AddOtOpcUaConfigDb(builder.Configuration);
|
||||||
|
builder.Services.AddOtOpcUaCluster(builder.Configuration);
|
||||||
|
|
||||||
|
// Akka cluster bootstrap. Role-specific singletons are registered on the AkkaConfigurationBuilder
|
||||||
|
// from inside the configurator lambda. AddAkka spins the ActorSystem at host start.
|
||||||
|
builder.Services.AddAkka("otopcua", (ab, _) =>
|
||||||
|
{
|
||||||
|
if (hasAdmin)
|
||||||
|
ab.WithOtOpcUaControlPlaneSingletons();
|
||||||
|
// Driver-role startup (DriverHostActor spawn + child probes) is wired in F19 once a
|
||||||
|
// RuntimeStartup contract is added — the actor itself exists (Phase 6), the registration
|
||||||
|
// extension does not yet. Without it, driver-role nodes still join the cluster and serve
|
||||||
|
// health/redundancy traffic but won't auto-spawn DriverHostActor.
|
||||||
|
});
|
||||||
|
|
||||||
|
if (hasAdmin)
|
||||||
|
{
|
||||||
|
// Auth + AdminUI surface only mounted on admin-role nodes. Driver-only nodes have no UI.
|
||||||
|
builder.Services.AddOtOpcUaAuth(builder.Configuration);
|
||||||
|
builder.Services.AddAdminUI();
|
||||||
|
builder.Services.AddSignalR();
|
||||||
|
builder.Services.AddOtOpcUaAdminClients();
|
||||||
|
}
|
||||||
|
|
||||||
|
builder.Services.AddOtOpcUaHealth();
|
||||||
|
|
||||||
var app = builder.Build();
|
var app = builder.Build();
|
||||||
app.MapGet("/", () => "OtOpcUa.Host scaffold");
|
app.UseSerilogRequestLogging();
|
||||||
|
|
||||||
|
if (hasAdmin)
|
||||||
|
{
|
||||||
|
app.UseAuthentication();
|
||||||
|
app.UseAuthorization();
|
||||||
|
app.UseAntiforgery();
|
||||||
|
app.MapOtOpcUaAuth();
|
||||||
|
app.MapAdminUI<App>();
|
||||||
|
app.MapOtOpcUaHubs();
|
||||||
|
}
|
||||||
|
|
||||||
|
app.MapOtOpcUaHealth();
|
||||||
|
|
||||||
|
Log.Information("OtOpcUa.Host starting with roles=[{Roles}] (admin={HasAdmin}, driver={HasDriver})",
|
||||||
|
string.Join(",", roles), hasAdmin, hasDriver);
|
||||||
|
|
||||||
await app.RunAsync();
|
await app.RunAsync();
|
||||||
|
|
||||||
|
namespace ZB.MOM.WW.OtOpcUa.Host
|
||||||
|
{
|
||||||
|
/// <summary>Re-exported for <c>WebApplicationFactory<Program></c> integration tests (F1).</summary>
|
||||||
|
public partial class Program;
|
||||||
|
}
|
||||||
|
|||||||
12
src/Server/ZB.MOM.WW.OtOpcUa.Host/Routes.razor
Normal file
12
src/Server/ZB.MOM.WW.OtOpcUa.Host/Routes.razor
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
@using ZB.MOM.WW.OtOpcUa.AdminUI
|
||||||
|
|
||||||
|
<Router AppAssembly="@typeof(EndpointRouteBuilderExtensions).Assembly">
|
||||||
|
<Found Context="routeData">
|
||||||
|
<RouteView RouteData="@routeData" DefaultLayout="@typeof(MainLayout)" />
|
||||||
|
</Found>
|
||||||
|
<NotFound>
|
||||||
|
<LayoutView Layout="@typeof(MainLayout)">
|
||||||
|
<p role="alert">Page not found.</p>
|
||||||
|
</LayoutView>
|
||||||
|
</NotFound>
|
||||||
|
</Router>
|
||||||
5
src/Server/ZB.MOM.WW.OtOpcUa.Host/_Imports.razor
Normal file
5
src/Server/ZB.MOM.WW.OtOpcUa.Host/_Imports.razor
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
@using Microsoft.AspNetCore.Components
|
||||||
|
@using Microsoft.AspNetCore.Components.Routing
|
||||||
|
@using Microsoft.AspNetCore.Components.Web
|
||||||
|
@using static Microsoft.AspNetCore.Components.Web.RenderMode
|
||||||
|
@using ZB.MOM.WW.OtOpcUa.Host
|
||||||
Reference in New Issue
Block a user