177 KiB
Frontend Primitives Contract
Contract Metadata
{
"subsystem_id": "frontend-primitives",
"lane": "L8",
"contract_file": "docs/release-control/v6/internal/subsystems/frontend-primitives.md",
"status_file": "docs/release-control/v6/internal/status.json",
"registry_file": "docs/release-control/v6/internal/subsystems/registry.json",
"dependency_subsystem_ids": [
"agent-lifecycle",
"cloud-paid",
"storage-recovery"
]
}
Purpose
Own reusable frontend primitives and canonical page-shell patterns so feature work extends shared components instead of creating new local variants.
Canonical Files
frontend-modern/src/components/shared/frontend-modern/src/components/Settings/Settings.tsxfrontend-modern/src/components/Settings/SettingsDialogs.tsxfrontend-modern/src/components/Settings/SettingsPageShell.tsxfrontend-modern/src/components/Settings/settingsPanelRegistry.tsfrontend-modern/src/components/Settings/APIAccessPanel.tsxfrontend-modern/src/components/Settings/AIChatMaintenanceSection.tsxfrontend-modern/src/components/Settings/AIModelSelectionSection.tsxfrontend-modern/src/components/Settings/AIProviderConfigurationSection.tsxfrontend-modern/src/components/Settings/AIRuntimeControlsSection.tsxfrontend-modern/src/components/Settings/AISettings.tsxfrontend-modern/src/components/Settings/AISettingsDialogs.tsxfrontend-modern/src/components/Settings/AISettingsStatusAndActions.tsxfrontend-modern/src/components/Settings/aiSettingsModel.tsfrontend-modern/src/components/Settings/AuditLogPanel.tsxfrontend-modern/src/components/Settings/useAuditLogPanelState.tsfrontend-modern/src/components/Settings/AuditWebhookPanel.tsxfrontend-modern/src/components/Settings/useAuditWebhookPanelState.tsfrontend-modern/src/components/Settings/CopyCommandBlock.tsxfrontend-modern/src/components/Settings/diagnosticsModel.tsfrontend-modern/src/components/Settings/DiagnosticsPanel.tsxfrontend-modern/src/components/Settings/DiagnosticsResultsPanel.tsxfrontend-modern/src/components/Settings/OperationsPanel.tsxfrontend-modern/src/utils/diagnosticsPresentation.tsfrontend-modern/src/utils/discoveryPresentation.tsfrontend-modern/src/components/Settings/GeneralSettingsPanel.tsxfrontend-modern/src/components/Settings/NetworkSettingsPanel.tsxfrontend-modern/src/components/Settings/RecoverySettingsPanel.tsxfrontend-modern/src/components/Settings/SecurityAuthPanel.tsxfrontend-modern/src/components/Settings/SecurityOverviewPanel.tsxfrontend-modern/src/components/Settings/settingsHeaderMeta.tsfrontend-modern/src/components/Settings/selfHostedBillingPresentation.tsfrontend-modern/src/components/Settings/SSOProvidersPanel.tsxfrontend-modern/src/components/Settings/useAISettingsState.tsfrontend-modern/src/components/Settings/useDiagnosticsPanelState.tsfrontend-modern/src/components/Settings/useSettingsShellState.tsfrontend-modern/src/components/Settings/useSSOProvidersState.tsfrontend-modern/src/components/Settings/ssoProvidersModel.tsfrontend-modern/src/utils/ssoProviderPresentation.tsfrontend-modern/src/utils/systemSettingsPresentation.tsfrontend-modern/src/utils/aiSettingsPresentation.tsfrontend-modern/src/utils/settingsShellPresentation.tsfrontend-modern/src/utils/textPresentation.tsfrontend-modern/src/components/Settings/UpdateInstallGuide.tsxfrontend-modern/src/components/Settings/updatesSettingsModel.tsfrontend-modern/src/components/Settings/UpdatesSettingsPanel.tsxfrontend-modern/src/components/Settings/ReportingPanel.tsxfrontend-modern/src/components/Settings/reportingPanelModel.tsfrontend-modern/src/components/Settings/reportingInventoryExportModel.tsfrontend-modern/src/components/Settings/useReportingPanelState.tsfrontend-modern/src/utils/reportingPresentation.tsfrontend-modern/src/utils/updatesPresentation.tsfrontend-modern/src/components/Settings/__tests__/settingsArchitecture.test.tstests/integration/tests/15-settings-shell-consistency.spec.tsfrontend-modern/src/components/shared/PageControls.guardrails.test.tsfrontend-modern/src/components/shared/TypeColumn.guardrails.test.tsfrontend-modern/src/features/frontend-modern/src/components/SetupWizard/SetupWizard.tsxfrontend-modern/src/components/Settings/useSettingsInfrastructurePanelProps.tsfrontend-modern/src/components/SetupWizard/SetupCompletionPreview.tsxfrontend-modern/src/components/SetupWizard/steps/WelcomeStep.tsxfrontend-modern/src/components/SetupWizard/__tests__/SetupWizard.test.tsxfrontend-modern/src/components/SetupWizard/__tests__/SetupCompletionPreview.test.tsxfrontend-modern/src/components/SetupWizard/__tests__/WelcomeStep.test.tsxfrontend-modern/src/components/shared/MonitoredSystemLimitWarningBanner.tsxfrontend-modern/src/components/Settings/SystemLogsPanel.tsxfrontend-modern/src/components/Settings/useSystemLogsPanelState.tsfrontend-modern/src/utils/systemLogsPresentation.tsfrontend-modern/src/components/Settings/__tests__/SystemLogsPanel.test.tsxfrontend-modern/src/pages/Operations.tsxfrontend-modern/src/components/Settings/ResourcePicker.tsxfrontend-modern/src/components/Settings/reportingResourceTypes.tsfrontend-modern/src/utils/reportableResourceTypes.tsfrontend-modern/src/utils/reportingResourceTypes.tsfrontend-modern/src/utils/problemResourcePresentation.tsfrontend-modern/src/utils/dashboardEmptyStatePresentation.tsfrontend-modern/src/utils/dashboardGuestPresentation.tsfrontend-modern/src/utils/dashboardKpiPresentation.tsfrontend-modern/src/utils/dashboardTrendPresentation.tsfrontend-modern/src/components/Toast/Toast.tsxfrontend-modern/src/utils/toast.tsfrontend-modern/src/utils/semanticTonePresentation.tsfrontend-modern/src/utils/emptyStatePresentation.tsfrontend-modern/src/utils/typeColumnPresentation.tsfrontend-modern/src/pages/__tests__/Operations.helpers.test.tsfrontend-modern/src/components/Settings/NetworkBoundarySettingsSection.tsxfrontend-modern/src/components/Settings/networkSettingsModel.tsfrontend-modern/src/components/Settings/useDiscoverySettingsState.tsfrontend-modern/src/components/Settings/useSettingsInfrastructurePanelProps.tsfrontend-modern/src/components/Settings/settingsPanelRegistryContext.tsxfrontend-modern/src/components/Settings/settingsPanelRegistryLoaders.tsfrontend-modern/src/components/Settings/settingsNavigationModel.tsfrontend-modern/src/components/Settings/settingsNavCatalog.tsfrontend-modern/src/components/Settings/settingsNavVisibility.tsfrontend-modern/src/components/Settings/settingsRouting.tsfrontend-modern/src/components/Settings/settingsTabSaveBehavior.tsfrontend-modern/src/components/Settings/settingsTypes.tsfrontend-modern/src/components/Settings/useSettingsNavigation.tsfrontend-modern/src/components/Settings/useSettingsPanelRegistry.tsxfrontend-modern/src/components/Settings/useSettingsSystemPanels.tsxfrontend-modern/src/components/Settings/DockerRuntimeSettingsCard.tsxfrontend-modern/src/components/shared/EnvironmentLockBadge.tsxfrontend-modern/src/utils/environmentLockPresentation.tsfrontend-modern/src/utils/docsLinks.tstests/integration/tests/20-local-doc-links.spec.tsfrontend-modern/src/index.cssfrontend-modern/src/components/shared/summaryInteractionA11y.tsfrontend-modern/src/components/shared/SummaryRowActionButton.tsxfrontend-modern/src/hooks/createNonSuspendingQuery.tsfrontend-modern/src/components/shared/SummaryTableCardHeader.tsxfrontend-modern/src/components/shared/UpgradeLink.tsxfrontend-modern/src/components/shared/useUpgradeNavigation.tsfrontend-modern/src/utils/upgradeNavigation.tsfrontend-modern/src/components/DemoBanner.tsxfrontend-modern/src/components/Login.tsxfrontend-modern/src/stores/demoMode.tsfrontend-modern/src/stores/sessionCapabilities.tsfrontend-modern/src/stores/sessionPresentationPolicy.tsfrontend-modern/src/stores/licenseCommercial.tsfrontend-modern/src/useAppRuntimeState.tsfrontend-modern/src/stores/aiChat.tsfrontend-modern/scripts/header-audit.mjs
Shared Boundaries
frontend-modern/src/components/Settings/GeneralSettingsPanel.tsxshared withsecurity-privacy: the general settings privacy panel is both a security/privacy control surface and a canonical settings-shell presentation boundary.frontend-modern/src/components/Settings/SecurityAuthPanel.tsxshared withsecurity-privacy: the authentication settings surface is both a security/privacy control surface and a canonical settings-shell presentation boundary.frontend-modern/src/components/Settings/SecurityOverviewPanel.tsxshared withsecurity-privacy: the security overview settings surface is both a security/privacy control surface and a canonical settings-shell presentation boundary.frontend-modern/src/stores/aiChat.tsshared withai-runtime: the assistant drawer and session store is both an AI runtime control surface and a canonical app-shell presentation boundary.
Extension Points
- Add shared primitives in
frontend-modern/src/components/shared/Shared monitored-system warning primitives under that path must stay compact app-shell pointers into the owned Pulse Pro billing surface. The shared banner may announce posture and route to the relevant billing tab, but durable plan-capacity explanation, over-plan reasoning, and upgrade/review actions belong in thecloud-paidplan surface rather than permanent banner-local prose. - Route new top-level settings surfaces through the canonical settings shell
instead of introducing page-local framing.
Shared shells and primitives that need websocket or dark-mode context must
consume
frontend-modern/src/contexts/appRuntime.ts; they must not importfrontend-modern/src/App.tsx, becauseApp.tsxowns provider placement while frontend primitives own reusable consumption. That same shared shell boundary now also owns thin public-route handoff presentation infrontend-modern/src/App.tsx: compatibility routes such as/pricingmay stay outside authenticated chrome, but they must remain minimal handoff shells that defer destination truth to the owning subsystem instead of embedding a second copy of public marketing or checkout UI inside the product runtime. The same settings-shell boundary owns read-only landing posture: when the session presentation policy says the operator cannot manage setup,/settingsand sidebar navigation must land on the canonical reporting/control surface instead of setup-oriented install routes. That same settings-shell boundary also owns explicit organization-route stability. Deep links such as/settings/organization,/settings/organization/access, and adjacent organization shells must keep their canonical header and page frame when the route itself is allowed, even while runtime capabilities or presentation policy are still settling. Shared shell filtering may hide the sidebar item until the governing state resolves, butsettingsHeaderMeta.ts,useSettingsAccess.ts, and the canonical settings-shell tests must not bounce an allowed organization route back toInfrastructurejust because nav filtering has not yet surfaced that tab. That same settings-shell boundary also owns authenticated operator identity propagation into organization panels.frontend-modern/src/components/Settings/settingsPanelRegistryContext.tsxmust derive the effective username from the resolved security status (proxyAuthUsername, thenssoSessionUsername, thenauthUsername) and pass that identity into the organization overview, access, and sharing panels so deep-linked organization routes do not collapse into anonymous read-only shells under proxy-auth, SSO, or local-auth sessions. That same shared session-presentation boundary also owns alerts read-only posture:/alertsmay continue exposing reporting tabs such as overview and history, but activation controls plus configuration routes must collapse out of the public-demo shell instead of advertising blocked management actions. That same public-demo presentation boundary also owns Settings support posture: the authenticated demo shell must not advertiseDiagnostics & Health,Data & Reports, orSystem Logsin the Settings navigation, and legacy/operations/*links must resolve through the canonical Settings routing boundary instead of reviving a standalone Operations utility tab or route-local support shell. Shared sparkline primitives must also stay CSP-safe by construction:frontend-modern/src/components/shared/InteractiveSparkline.tsxmay use SVG attributes and shared state/model helpers for cursor, axis-label, and tooltip positioning, but it must not write inlinestyle=attributes for tick labels, tooltip placement, or per-series transitions on the public shell. Axis labels must render in fixed-size SVG shells or another non-scaled primitive boundary; the shared sparkline must not put axis glyphs insidepreserveAspectRatio="none"label viewBoxes that stretch text as the chart resizes. The same shared presentation boundary also owns reusable scroll containers:frontend-modern/src/components/shared/Table.tsxmust keep touch-scroll behavior on classes and shared CSS infrontend-modern/src/index.cssinstead of reintroducing inlinestyle=attributes for overflow or mobile scroll behavior.frontend-modern/src/components/shared/PulseDataGrid.tsxinherits that same boundary: shared data-grid shells must route scrollbar hiding and table-width sizing through shared classes plus HTML attributes, not inline overflow or min-width styles. - Add feature-specific presentation only when no shared primitive should own it
- Add guardrail tests when a new shared pattern is introduced.
Shared monitored-system warning primitives must prove their admission-freeze
posture through the canonical
frontend-modern/src/utils/monitoredSystemPresentation.tshelper plus runtimemonitored_system_capacityreads rather than reconstructing rawcurrent / limitslash math or0 remainingcopy in the banner shell, state owner, or shared model. Shared modal scroll containment follows that same owner split. The dialog shell infrontend-modern/src/components/shared/dialogModel.tsmust keep shared panelsmin-h-0, and page-owned modal bodies may useoverflow-y-autoonly under shrinkable flex columns instead of clipping lower fields behind a fixed-height shell. Shared filter popovers follow the same primitive-level ownership. The sharedFilterToolbarpanel class must render above nested card, table, and empty-state shells, and feature pages embedding those controls must make only their immediate control shell overflow-visible rather than forking local z-index or popover positioning rules. The shared navigation guide owns route-aware first focus: when it opens from a top-level product route such as/recovery, the first highlighted step should match that route instead of always restarting at Dashboard. - Keep shared infrastructure shell state on the reusable settings boundary:
frontend-modern/src/components/Settings/useSettingsInfrastructurePanelProps.tsandfrontend-modern/src/components/Settings/InfrastructureWorkspace.tsxmust continue to derive provider counts, availability, and shared subtab copy from one infrastructure-settings source — via the unified aggregator throughfrontend-modern/src/components/Settings/useConnectionsLedger.ts— instead of creating provider-local summary fetches or VMware-only shell vocabulary. Phase 9 retired the oldPlatformConnectionsWorkspaceper-type shell, but setup guidance may still usePlatform connectionsas the operator-facing label for the shared API-backed onboarding path. That same shared shell boundary now owns the first-run posture for/settings/infrastructure: the landing route should read as one source-manager workspace with configured infrastructure instances first and no redundant monitored-systems ledger beneath it. The landing route may include a compact guidance strip that explains platform APIs and host agents as Pulse 6 infrastructure sources and exposesDetect from address,Install Pulse Agent, andChoose source typeas first-run actions. It may also show a compact readiness strip derived from the same unified connection rows and discovered candidates so operators can confirm connected-system count, API coverage, agent coverage, discovery review state, and the next setup action without opening a tour or second ledger. Existing sources stay visible on the page, and add, detect, install, review, and edit flows open as secondary interactions from that same destination instead of taking over the whole page. Those secondary views must stay under the same singleInfrastructuresidebar destination, but they may open in governed modal/dialog chrome when that preserves the persistent source-manager page behind them. That governed dialog chrome must also preserve inner form scrolling:InfrastructureWorkspace.tsxandConnectionEditor.tsxkeep the add/edit shell onmin-h-0flex columns so long credential forms scroll inside the modal body instead of trapping the lower fields below the fold. The same shared shell boundary now also owns grouped source-row composition.useConnectionsLedger.ts,InfrastructureSourceManager.tsx, andInfrastructureWorkspace.tsxmust render attached collection methods as a plain-language row subtitle on the owning row (via platform API,via Pulse Agent, orvia platform API and Pulse Agent), with fuller detail in the edit dialog, instead of duplicating the same machine across multiple peer groups or forcing operators to decode badge jargon. That same shared shell boundary owns the landing taxonomy too: the primary grouping labels in the infrastructure manager must describe real platform/system owners, not collection methods. Agent-only machines belong in a standalone-host bucket, whilePulse Agentremains a collection- method label, install path, and detail-surface concept rather than a peer top-level pseudo-platform beside Proxmox, VMware, and TrueNAS. That same shared shell boundary also owns compact version visibility for agent-backed rows. The infrastructure source table must not grow a dedicated always-on version column for Pulse Agent; exact version text belongs in the edit/detail surfaces, while the landing table only surfaces a compact warning badge when an attached or standalone agent actually has an update available. That same table boundary must reuse the existingSystemandEndpointcells for compact standalone-agent identity such asUnraid 7.1.0plus a reported host address; it must not add a new diagnostics column just to surface host facts the unified agent already reports. That same shared shell boundary now owns one canonical infrastructure destination in the Settings sidebar.InfrastructureWorkspace.tsxowns the source-manager landing inside that destination, while route-backed add flows and local edit flows stay single-purpose instead of stacking multiple page-level workspaces at once. The source-manager landing now also owns the explicit discovery strip for that destination.InfrastructureSourceManager.tsxmay expose a compact discovery status line plusRun discoveryandDiscovery settingsactions from the shared landing shell, but it must not start a network scan just because the page rendered. New-source admission belongs on the table's per-platformAddactions or the compact first-run/readiness actions rather than in the discovery strip, and the direct address-probe utility may appear as first-run setup guidance while header discovery actions remain dedicated to saved network scanning. Discovered API-backed candidates stay visible in the same platform-group table as configured sources, using the existing tree/table hierarchy instead of spawning a second discovery-only page or card stack.InfrastructureWorkspace.tsxmust still open a new connection throughfrontend-modern/src/components/Settings/ConnectionEditor/ConnectionEditor.tsx, but the editor now serves as governed dialog content under the source manager rather than replacing the page inline. The?add=pickroute owns the grouped source-type picker,?add=detectowns the detect-from-address utility, and typed add routes jump straight into the matching credential slot throughinitialType. Credential slots are dispatched by the detected or manually-selected type and must still reach the canonical form body rather than diverging into a revived provider-specific workspace. For PVE, PBS, and PMG, the credential slot isfrontend-modern/src/components/Settings/ConnectionEditor/CredentialSlots/NodeCredentialSlot.tsxand it must composeNodeModalBasicInfoSection,NodeModalAuthenticationSection,NodeModalMonitoringSection, andNodeModalStatusFooterinline under the editor shell rather than embedding the full Proxmox workspace (discovery card, configured nodes table, delete dialog, node modal stack). For TrueNAS and VMware the credential slots arefrontend-modern/src/components/Settings/ConnectionEditor/CredentialSlots/TrueNASCredentialSlot.tsxandfrontend-modern/src/components/Settings/ConnectionEditor/CredentialSlots/VMwareCredentialSlot.tsxand they must render only the connection form body inline under the editor shell — no connection list, no row actions, no surrounding panel chrome. Showing a ledger of other systems inside the credential slot is exactly the ledger-inside-editor drift this contract forbids. The configured-connections summary and source-manager rows themselves must render exclusively from the aggregator.InfrastructureWorkspace.tsxcomposes the platform-banded systems table fromfrontend-modern/src/components/Settings/useConnectionsLedger.ts(pollingGET /api/connections). Table rows may open a governed edit dialog for mutable sources, but pause, resume, remove, last-error detail, and agent uninstall commands must live inside that owned edit surface or the shared row-action owner rather than returning to inline landing-page action clutter or a revived provider-specific detail page. When the backend marks a grouped Proxmox row with canonical cluster identity, the table primitive must render that cluster moniker as the row title instead of falling back to one sibling node hostname or reopening a standalone-host presentation for cluster-member agents. When that grouped row also carries backend-authored cluster members, the table primitive must render those nodes as child composition beneath the cluster row rather than flattening them back into peer top-level systems or hiding them entirely. That same landing-shell boundary also owns represented-host dedupe between the unified ledger and the discovery strip.InfrastructureWorkspace.tsx,frontend-modern/src/components/Settings/useConnectionsLedger.ts, andfrontend-modern/src/components/Settings/infrastructureSettingsModel.tsmust treat backend-authored hostname/IP aliases as canonical identity so an already-represented platform row, attached agent augmentation, or grouped member suppresses the matching discovered candidate instead of showing the same machine twice under hostname-versus-IP drift. Phase 9 retired the parallel reporting/inventory surface entirely:useInfrastructureReportingState,InfrastructureOperationsController,InfrastructureInventorySection,InfrastructureActiveRowDetails,InfrastructureIgnoredRowDetails,InfrastructureStopMonitoringDialog, and the per-type shellsPlatformConnectionsWorkspace,ProxmoxSettingsPanel,ProxmoxDirectWorkspace,NodeModal.tsx,TrueNASSettingsPanel, andVMwareSettingsPanelno longer exist. The aggregator plusConnectionEditoris the only path; no parallel reporting state, stop-surface dialog, ignored-row fallback, or per-type workspace may be reintroduced.connectionsTableModel.tscarries only the connection-scopedSystemManageActionvariant —inventory-active/inventory-ignoredmanage kinds must not return. - Keep Proxmox deep-link route selection on the shared settings-navigation boundary.
frontend-modern/src/components/Settings/settingsNavigationModel.tsandfrontend-modern/src/components/Settings/useSettingsNavigation.tsmust treat the canonical PBS and PMG Proxmox deep links as agent-selection authority even though those URLs resolve to the sharedinfrastructure-operationstab. Reloading or remounting on a PBS or PMG deep link must not silently fall back to the PVE selector state. - Keep shared storage feature presenters on canonical platform truth. When reusable storage presenters under
frontend-modern/src/features/storageBackups/classify canonical resources for the shared storage route, API-backed virtualization datastores such as VMware must stay inventory-only datastores instead of inheriting PBS-specific backup-repository or protected-target copy from older fallback branches. - Keep shared source/platform vocabulary on the governed manifest boundary.
frontend-modern/src/utils/platformSupportManifest.generated.tsmust be the tracked frontend projection ofdocs/release-control/v6/internal/PLATFORM_SUPPORT_MANIFEST.json,frontend-modern/src/utils/platformSupportManifest.ts,frontend-modern/src/utils/sourcePlatforms.ts, andfrontend-modern/src/utils/sourcePlatformOptions.tsmust consume that generated projection instead of embedding divergent future-label lists, setup/onboarding path allowlists, or presentation-only guesses, andfrontend-modern/scripts/canonical-platform-audit.mjsmust fail when the generated projection drifts from the governed manifest. - Keep top-of-page summary interaction on shared primitives. Infrastructure, workloads, and storage summary cards must route sticky-shell behavior through
frontend-modern/src/components/shared/StickySummarySection.tsxand route row-hover or focused-series rendering through shared chart primitives such asfrontend-modern/src/components/shared/InteractiveSparkline.tsxandfrontend-modern/src/components/shared/DensityMap.tsx, rather than page-local sticky wrappers or metric-card-specific hover logic. The shared summary-card contract must also own stable summary-card geometry for chart-backed cards so row hover, focus, synchronized readouts, or idle header metadata cannot ratchet the sticky summary taller across rerenders. - Keep summary chart interaction identity on one shared helper. Summary surfaces that expose row-hover, group-hover, chart-hover, or route-focus-driven chart emphasis must derive page/group/entity scope through
frontend-modern/src/components/shared/summaryCardInteraction.tsand pass that same resolved scope into card-state, sparkline, and density-map primitives, rather than letting cards readhovered || focusedwhile charts listen to a different page-local ID source. Hovering one summary chart must promote that series into the shared active entity so sibling cards highlight the same object instead of keeping chart-local hover islands, and hovering or pinning a workload group header, infrastructure cluster header, or storage pool-group header must scope the matching summary cards through that same shared contract instead of forking a page-local summary filter path. Sibling cards should surface that synchronized hover as one compact header readout through the shared summary-card contract, while the chart under the pointer keeps the only floating tooltip.frontend-modern/src/components/Recovery/RecoverySummary.tsxis explicitly outside this interaction dialect: recovery posture cards may share summary framing, but they must not silently grow row/group/chart hover behavior without a separate governed product decision. - Keep page summaries page-scoped when table rows enter contextual focus. Route-backed row selection may add a focused label and shared series emphasis, but infrastructure, workloads, and storage summary cards must continue to render the page-level series set instead of collapsing the summary down to the selected row or replacing the global trend view with row-local empty states.
- Keep contextual row focus on the shared summary primitive. Summary surfaces and same-route table drill-ins must reuse
frontend-modern/src/components/shared/contextualFocus.tsfor interactive-series filtering, focused-name lookup, active-series derivation, local scroll preservation, and deliberate inline-detail reveal instead of rebuilding page-localSetfilters, focused-label scans, drawer-aware scroll math, or ad hoc scroll restoration in each surface. - Keep summary-to-table coordination deliberate, explicit, and reversible. Shared summary hover may highlight the matching table row when it is already visible, but transient chart hover must not auto-filter tables, auto-scroll the page, or reshuffle table ordering. Pinned page/group/entity scope on workloads, infrastructure, or storage must stay row-first: the pinned row or group header is the visible scoped state, not a second strip or search-row widget. Page shells therefore must not reintroduce always-on scope banners, preview bars, page-local chips, breadcrumbs, or search/filter-row scope accessories just to explain pinned state. When the active row is off-screen, page owners must still route through
frontend-modern/src/components/shared/summaryTableFocus.tsand surface a lightweightJump to rowaffordance that reveals and scrolls only on explicit user action. That same shared table-focus owner now also owns reversible clearing: pinned scope may clear only from governed neutral interaction-surface space or the sharedEscapepath, with page owners binding a broader clear-surface root separately from the row-lookup table root when needed and supplying one page-level reset callback for filters plus summary-linked selections. Row cells, group headers, inline detail, summary cards, and explicit controls must not accidentally clear pinned scope, while governed table/card clear surfaces must still allow real user clicks on neutral whitespace to clear it. Deliberate row focus may reveal inline detail automatically, but that reveal must be drawer-aware: infrastructure and workload row toggles that already have the row in view must hand the current.app-scroll-shellposition throughfrontend-modern/src/utils/appShellScrollRestoration.tsso the remounted shell infrontend-modern/src/App.tsxcan reopen the inline detail without looking like a page refresh, and then still route through the shared reveal helper whenever the opened drawer would otherwise land below the fold. Same-route drawers must therefore scroll only enough to keep the row header plus the top of the inline detail visible, never hard-center the row just because the route state changed. Shared summary-linked rows and group headers must also route their preview semantics throughfrontend-modern/src/components/shared/summaryInteractionA11y.ts. Leaf rows and any explicit row-level control chrome must route deliberate pin/open ownership throughfrontend-modern/src/components/shared/SummaryRowActionButton.tsx, soaria-expanded,aria-controls,aria-pressed, focus treatment, andEscapepreview clearing stay on the shared control instead of focusable- table-row shims. Group headers are different: they may use the header row itself as the deliberate pin target when that keeps the table chrome native, but they must not grow separate scope/pinned pill buttons or off-screen fallback strips. Workloads, infrastructure, and storage must not rebuild row-as-button keyboard handling or trailing one-off expand columns once the shared action primitive exists. When pinned page, group, or entity scope needs a local explicit reset, the only shared table-chrome owner isfrontend-modern/src/components/shared/SummaryTableCardHeader.tsx: the reset action stays as one compact header-levelClearcontrol with an accessibleClear selectionlabel, not a second page-level scope strip, search-row accessory, or filter-bar badge. - Keep summary-linked table row emphasis on the shared primitive contract. Workloads, infrastructure, and storage rows that mirror the active summary entity must expose that state through
data-summary-row-activeand let the shared presentation infrontend-modern/src/index.cssrender the row emphasis, rather than carrying page-local sky or blue fill classes inside each row renderer. Group-scoped preview and pin must use that same shared presentation boundary: child rows that belong to a hovered or pinned summary group should exposedata-summary-group-member-active="preview|pinned"so the block-level emphasis stays subtle, consistent, and reversible instead of each table inventing its own outline, badge, or full-strength fill treatment. Storage-backed reusable row presenters underfrontend-modern/src/features/storageBackups/must also keep row height and alert accents on class/data-attribute presentation instead of runtime inline style maps, so the shared table contract stays CSP-safe on both steady-state and alert-highlighted routes. - Keep retained-value data loading honest at the ownership boundary. Helpers
that prevent a feature surface from falling through the app-level Suspense
boundary during in-flight refresh should stay feature-local until multiple
governed surfaces truly share the behavior. Once that boundary is shared,
promote the helper into an explicit shared hook owner such as
frontend-modern/src/hooks/createNonSuspendingQuery.tsrather than re-copying suspense-escape logic into each feature area or burying it inside one feature's private state model. - Keep shared commercial warning banners truthful about destination intent.
When a shared banner renders both explanatory and commercial CTAs, those
labels must resolve to distinct owned destinations or section anchors
instead of presenting two different labels that land on the same
unscoped billing screen. Monitored-system warning banners must also honor
canonical runtime usage availability before treating a limit as urgent:
when the backend marks
max_monitored_systems.current_available=false, the shared banner model must consume the cloud-paid monitored-system presentation helper and suppress usage summaries, upgrade pressure, and upgrade-impression telemetry rather than rendering stalecurrent/limitcounts or paid-plan CTAs from banner-local availability checks. When the banner does need an upgrade destination, it must scope the operator into the cloud-paid plan-selection intent (intent=self_hosted_plan) rather than reintroducing browser symbols or CTA copy that frame the flow as monitored-system-cap expansion. - Keep assistant availability bootstrap on the shared app-shell boundary.
frontend-modern/src/useAppRuntimeState.ts,frontend-modern/src/App.tsx,frontend-modern/src/stores/aiChat.ts,frontend-modern/src/components/AI/Chat/index.tsx, andfrontend-modern/src/hooks/useDashboardActions.tsmust consume the backend-owned/api/security/status.sessionCapabilities.assistantEnabledfact instead of probing/api/settings/aior/api/ai/*during ordinary route bootstrap. Closed assistant chrome and non-AI settings panels may not initialize assistant runtime state until an owned assistant or Patrol surface is actually open.frontend-modern/src/stores/aiChat.tsis the shared drawer shell owner for assistant open/close state, focus handoff, and tenant-local context/session persistence; the app shell must not fork that state acrossApp.tsx,AppLayout.tsx, or page-level helpers. The same shared shell boundary must keep Pulse Assistant coherent while a blocking shared dialog owns the viewport: closed launcher affordances must hide until the dialog clears, and the shell must close any already-open assistant drawer instead of leaving background assistant controls visibly active behind the modal. AI-owned frontend surfaces that need shared settings or model-catalog truth must route those reads throughfrontend-modern/src/stores/aiRuntimeState.tsrather than each feature bootstrapping/api/settings/aior/api/ai/modelsindependently. Non-AI settings panels such asfrontend-modern/src/components/Settings/useAgentProfilesPanelState.tsmust stay on the app-shell assistant-availability fact instead of re-reading raw AI settings just to decide whether assistant affordances should render. - Keep optional shared selectors honest about data ownership. Reusable
shells such as
frontend-modern/src/components/shared/InfrastructureSelector.tsxmust gate their runtime data hooks on actual surface visibility, passing explicit disabled/null inputs to shared data owners when the selector is hidden instead of hydrating background summary data for chrome the page is not rendering. - Keep Patrol shell composition and product-first provider vocabulary on the
shared feature-presentation boundary.
frontend-modern/src/features/patrol/PatrolIntelligenceSummary.tsx,frontend-modern/src/features/patrol/PatrolIntelligenceWorkspace.tsx,frontend-modern/src/features/patrol/PatrolIntelligenceHeader.tsx,frontend-modern/src/features/patrol/PatrolIntelligenceBanners.tsx,frontend-modern/src/features/patrol/usePatrolIntelligenceState.ts,frontend-modern/src/features/patrol/patrolInvestigationContextModel.ts,frontend-modern/src/features/patrol/patrolSupportingContextPresentation.ts, andfrontend-modern/src/components/patrol/PatrolStatusBar.tsxmust keep Patrol assessment, verification, and findings primary; surface recent changes, learned correlations, and policy coverage only as explicitly secondary supporting context when degraded or incomplete verification, active findings, or selected-run investigation makes that evidence relevant; and use Patrol/provider wording for the shared provider settings, provider model, and provider circuit-breaker affordances instead of generic AI labels inside Patrol-owned shells. The shared app shell infrontend-modern/src/App.tsxandfrontend-modern/src/AppLayout.tsxmust likewise expose/patrolas the canonical route and navigation target, keeping legacy/aientry points as thin compatibility redirects rather than a second Patrol-branded primary route. The Patrol-owned supporting-context presenter must also keep the disclosure toggle plus evidence-boundary copy centralized instead of letting the workspace reintroduce inline shell-local trust wording. - Keep the shared
system-aisettings shell product-first.frontend-modern/src/components/Settings/AISettings.tsx,frontend-modern/src/components/Settings/settingsHeaderMeta.ts,frontend-modern/src/components/Settings/settingsNavCatalog.ts,frontend-modern/src/components/Settings/useAISettingsState.ts, andfrontend-modern/src/utils/aiSettingsPresentation.tsmust present that surface to operators asAssistant & Patrolplus provider/model configuration rather than as a genericAI Servicesshell. Runtime controls insidefrontend-modern/src/components/Settings/AIRuntimeControlsSection.tsxmust likewise describe discovery as workload discovery that supplies concrete service context to Pulse Assistant and Patrol, not as a generic AI context feature. Assistant-only controls inside the shared shell, such as execution permissions and session maintenance, must stay explicitly labeled as Pulse Assistant controls, while Patrol schedule and autonomy continue to live on Patrol-owned surfaces rather than drifting back into the shared settings shell. Shared/default model choices may remain on the combined shell only when Assistant and Patrol overrides are presented as explicit per-surface overrides instead of a generic advanced AI bucket.
Forbidden Paths
- Reinventing table/filter/toggle primitives when a shared version exists
- Feature-local styling forks of canonical shared components without explicit justification
- Direct imports that bypass shared presentation helpers where guardrails exist
- Top-level settings panels introducing bespoke page-level headers or outer
framing instead of the canonical settings shell and
SettingsPanelcontract
Completion Obligations
- Update guardrail tests when new shared primitives are added
- Keep top-level settings surfaces routed through the canonical settings shell
and maintain both
frontend-modern/src/components/Settings/__tests__/settingsArchitecture.test.tsplustests/integration/tests/15-settings-shell-consistency.spec.ts - Update this contract when a new canonical UI pattern is adopted
- Remove local forks after the shared primitive is introduced
- Keep shared feature-level presenters on capability truth. When reusable
presenters under
frontend-modern/src/features/explain why a control, chart, or detail surface is unavailable, they must describe the owned identity or capability gap instead of prescribing a provider-local install path that conflicts with API-backed platforms like TrueNAS. - When a settings route header and a top-level settings shell describe the same
commercial surface, keep them on the same shared presentation owner instead
of allowing route metadata in
settingsHeaderMeta.tsor labels insettingsNavCatalog.tsto drift into independent title or description copy, and keep adjacent settings-shell referrals such asInfrastructureWorkspace.tsxon that same shared owner instead of reintroducing local “go to Pulse Pro” variants. That same shared owner must also support explicit IA/title separation for self-hosted commercial settings: the nav label may stay product-IA-first (Plans) while the page shell stays task-first (Plans & Activation), and the owned plan shell must foreground the active plan name plus unlocked capabilities before secondary billing or recovery detail so paid upgrades can confirm their entitlement immediately after activation. - Keep hosted settings-shell framing imports safe for bundle initialization.
Self-hosted billing titles, descriptions, and referral copy used by
settingsHeaderMeta.ts,settingsNavCatalog.ts, and adjacent settings shells must flow throughfrontend-modern/src/components/Settings/selfHostedBillingPresentation.tsinstead of importing generic commercial presentation helpers directly into hosted settings route shells. - Keep first-session dashboard empty-state copy on
frontend-modern/src/utils/dashboardEmptyStatePresentation.ts, and make infrastructure setup guidance name the canonical destination explicitly instead of falling back to generic settings CTA labels. - Keep the live first-session wizard on the canonical three-step runtime
shape in
frontend-modern/src/components/SetupWizard/SetupWizard.tsx(Welcome,Security, thenInstall), and keep the step indicator plus completion CTA language aligned with the governed infrastructure install workspace instead of regressing to a route jump that leaves the next action implicit. Preview-only follow-up surfaces such asfrontend-modern/src/components/SetupWizard/SetupCompletionPreview.tsxmust stay deterministic and scenario-driven: they may not poll the live/api/stateruntime or inherit whatever connected systems happen to exist on the current backend, and browser proof for/preview/setup-completemust select explicit preview scenarios instead of ambient runtime state. - Keep AI settings setup UI backend-driven:
frontend-modern/src/components/Settings/useAISettingsState.tsandfrontend-modern/src/components/Settings/AISettingsDialogs.tsxmay collect provider credentials or runtime URLs, but they must not bake vendor model IDs into setup payloads. The shared settings shell should let the backend resolve the effective BYOK model and then render that returned state rather than guessing a model in the modal. - Keep shared filter primitives coherent with route-owned option hydration.
Feature shells such as
frontend-modern/src/features/infrastructure/must keep a route-owned canonical option visible in shared selects likeLabeledFilterSelecteven when current results do not contain that option, so provider-scoped handoffs do not flash back toAll. - Keep the first welcome screen in
frontend-modern/src/components/SetupWizard/steps/WelcomeStep.tsxexplicit about operator context. The shell must explain that the bootstrap token only unlocks first-run setup, state where the command should run, and adapt command/help text to detected Docker or containerized deployments instead of assuming the operator already knows which host or container owns the Pulse install. - Keep the settings-shell infrastructure landing path aligned with that same
first-session story.
frontend-modern/src/components/Settings/settingsNavigationModel.tsmust treat/settingsand the infrastructure settings tab as the canonical path to the bare/settings/infrastructure, which renders the unified Connections table, not to a separate install subview or to reporting/ control. The first-session story is owned by that table's own empty state and theAdd connectionentry point on it, not by a second landing route, so first-time operators and returning operators see one consistent infrastructure surface by default. - Keep dashboard onboarding copy on the shared presentation owner in
frontend-modern/src/utils/dashboardEmptyStatePresentation.ts. Both the infrastructure empty state and the dashboard route's no-resources state must name the canonical install workspace explicitly, keepPlatform connectionsvisible as the API-backed alternative for Proxmox and TrueNAS, and expose the same first-host next step instead of falling back to passive “nothing here yet” wording. - Keep cross-surface investigation handoffs on shared route ownership.
Feature shells such as Alerts and Patrol may decide which governed
destination chips to render, but canonical href, label, dedupe, and
infrastructure-fallback truth must stay in
frontend-modern/src/routing/resourceLinks.tsinstead of freezing raw route strings or provider-local link builders inside feature panels. - Keep shared summary-card emphasis coherent. When shared summary primitives enter an
inactivestate,SummaryMetricCard,InteractiveSparkline, andDensityMapmust all demote background context together so storage, infrastructure, and workloads read as one interaction model instead of mixing page-local opacity, sticky-shell, or highlight rules. - Keep density-map summaries overview-first. When a shared summary density map receives row focus or chart-hover emphasis,
frontend-modern/src/components/shared/DensityMap.tsx,frontend-modern/src/components/shared/useDensityMapState.ts, andfrontend-modern/src/components/shared/densityMapModel.tsmust preserve the multi-entity overview rows and keep focused-entity detail in the hover tooltip instead of swapping the card into a single-series chart, dimming the rest of the map into unusable background noise, duplicating cursor-value tooltip copy, or adding persistent card chrome that steals heatmap space. The card body must stay overview-first; the tooltip may carry the active entity identity, current value, and peak, shared tooltip shells must follow semantic surface tokens instead of forcing a dark palette in light mode, the tooltip header must let long entity names consume the available width before truncating rather than clipping against an arbitrary fixed label cap, numeric metric readouts such as16.9 MB/sor37.4 MB/smust stay single-line instead of wrapping the unit onto a second row, and density-map detail that cannot fit cleanly inside the canonical tooltip shell must be omitted rather than introducing tooltip-specific chrome or a secondary chart inside the hover surface. - Keep shared commercial and Patrol quickstart presenters on runtime-backed wording. Reusable shells and helper-driven badges must describe quickstart as Patrol runs or Patrol-only activation support when that is the governed backend truth, and must not drift back to generic hosted-chat or generic AI-credit claims.
- Keep sparkline scrubbing source-local and sibling-sync timestamp-based. The chart a user is actively scrubbing in
frontend-modern/src/components/shared/InteractiveSparkline.tsxandfrontend-modern/src/components/shared/useInteractiveSparklineState.tsmust keep its dashed hover cursor on the real local mousex, while sibling cards may map the shared hover timestamp onto their own timelines. Shared cursor sync must not snap the source chart back onto the nearest sample timestamp, the rendered SVG/canvas hover cursor must bind to the actual numeric cursor coordinate rather than a boolean guard state, the time cursor must span the chart viewport instead of collapsing to the series height, and the hover tooltip must track the pointer instead of anchoring to the chart top edge while following the active theme rather than a hardcoded dark shell. - Keep shared contextual focus canonical after adoption. Once a summary or table surface enters route-backed contextual focus, future additions must extend
frontend-modern/src/components/shared/contextualFocus.tsand its guardrail tests rather than forking another helper for workload IDs, resource IDs, or scroll-preserving same-route selection. - Keep shared infrastructure/resource selectors on the canonical agent-facet
truth. Shared primitives and settings-facing selector helpers must treat
top-level TrueNAS appliances as agent-facet infrastructure via shared
helper ownership instead of reviving a direct
resource.type === 'truenas'branch inside page shells, selectors, or reporting-resource type helpers. - Keep shared feature-shell Patrol run fixtures on the canonical run-record
contract. When
frontend-modern/src/features/patrol/consumes Patrol run history, the shared normalized record must preserve provider-backed counts such astruenas_checkedinstead of letting feature-local fixtures or fallback objects collapse API-backed TrueNAS systems back into generic agent-host presentation. That same shared route-shell boundary also owns header-composition audit.frontend-modern/scripts/header-audit.mjs,.github/workflows/release-dry-run.yml, and.github/workflows/create-release.ymlmust prove the same shared top-level page-header contract before publication. The audit may follow local imports when a route shell composesPageHeaderthrough a nested surface, and settings coverage must stay limited to top-level registry panels rather than every helper*Panel.tsxfile. The canonical Settings shell therefore owns the sharedPageHeaderfor support tools, andfrontend-modern/src/pages/Operations.tsxmust stay a redirect-only compatibility handoff instead of regrowing a second route-local heading, tab strip, or page shell for diagnostics, reporting, or logs. - Keep the authenticated app root aligned with that same first-session path.
That same shared-primitive ownership now includes contextual row focus.
frontend-modern/src/components/shared/contextualFocus.tsis the canonical owner for interactive-series filtering, focused-label lookup, active-series resolution, and nearest-scrollable-ancestor preservation across page-scoped summary surfaces. Dashboard row focus, infrastructure summary emphasis, storage summary emphasis, and workloads summary emphasis must all route through that helper instead of maintaining page-local copies of the same hover/focus rules.frontend-modern/src/App.tsxmust land/on the dashboard shell and let the governed dashboard empty state route first-time operators into Infrastructure Install, instead of preserving a separate root-only jump to/infrastructurethat drifts from the rest of the onboarding contract. The same entry-shell contract must also canonicalize authenticated/login: once auth succeeds, the shared shell must resolve that route back onto the governed dashboard landing path instead of rendering a page-local not-found state inside the authenticated chrome. - Keep relay settings shell copy on the shared presentation owner in
frontend-modern/src/utils/relayPresentation.ts. The route metadata insettingsHeaderMeta.tsand the leadingSettingsPanelinRelaySettingsPanel.tsxmust reuse the same description and availability copy instead of drifting into separate rollout or pairing wording. - Keep shared settings-shell legal and docs referrals on
frontend-modern/src/utils/docsLinks.ts. Shared settings surfaces such asAIRuntimeControlsSection.tsxmust not hardcode GitHubmaindoc URLs for privacy, security, proxy-auth, scope-reference, or Terms-of-Service links. - Keep shared settings-shell telemetry transparency controls on the governed
general settings panel. Preview/reset affordances for anonymous telemetry
must stay rendered inside
frontend-modern/src/components/Settings/GeneralSettingsPanel.tsxinstead of drifting into route-local modals, hidden dev tools, or shell chrome that operators would not naturally inspect. - Keep the short telemetry/privacy summary copy on that same shared surface
accurate to the governed privacy doc. If the trust boundary depends on a
specific retention window or on “IP addresses are not stored” rather than
“IPs are never seen,” the summary copy in
GeneralSettingsPanel.tsxmust state those facts plainly instead of reverting to a stronger but inaccurate shorthand. - Keep shared storage-route feature presentation on neutral capability truth.
Reusable mappers and presenters in
frontend-modern/src/features/storageBackups/must distinguish inventory datastores from backup repositories so VMware rows on the shared storage route stay canonical to the admitted phase-1 floor instead of reviving backup-target, protected-target, or recovery-local semantics on a shared page. - Keep infrastructure settings-shell API alternatives on the shared shell
contract.
frontend-modern/src/components/Settings/InfrastructureWorkspace.tsx,frontend-modern/src/components/Settings/settingsHeaderMeta.ts, andfrontend-modern/src/components/Settings/settingsNavigationModel.tsmust present the unified add flow as the canonical API-backed entry for Proxmox, TrueNAS, VMware, and future provider integrations instead of reviving top-levelDirect Proxmoxwording or shell-local provider routes. Phase 9 retired thePlatform connectionsnomenclature along with the shells that owned it — there is noPlatformConnectionsWorkspaceand no per-typeProxmoxSettingsPanel/TrueNASSettingsPanel/VMwareSettingsPanelto route through; the provider is a field inside oneConnectionEditor, not a destination. - Keep the infrastructure settings connection inventory on one shared
source.
frontend-modern/src/components/Settings/InfrastructureWorkspace.tsxcomposes rows exclusively fromfrontend-modern/src/components/Settings/useConnectionsLedger.ts, which pollsGET /api/connections. Provider connection counts and availability must derive from that aggregator, not from a top-level ledger plus parallel provider-specific fetches. The retiredPlatformConnectionsWorkspace/TrueNASSettingsPanel/VMwareSettingsPanelpanels must not be reintroduced as a second fetch path. - Keep alert-history feature composition on the current owned state contract.
frontend-modern/src/features/alerts/tabs/HistoryTab.tsxmust react to the sharedalertData()history state instead of reviving deleted aliases, and it must pass unified-resource resolution through tofrontend-modern/src/features/alerts/AlertResourceIncidentsPanel.tsxso the panel can render shared route chips without creating another page-local resource lookup or provider-specific handoff layer. - Keep the alert-thresholds containers surface on the canonical shared owner.
alertOverridesModel.ts,useAlertOverridesState.ts, anduseAlertsConfigurationState.tsmust surface API-backedapp-containerparents such as TrueNAS as first-classContainer Runtimes, whileThresholdsTab.tsxmust bridge function-valued selectors intoThresholdsTable.tsxexplicitly instead of relying on spread-based adapter props that can collapse functions on the live Solid surface. Docker-only controls inThresholdsTableDockerTab.tsxmust remain gated to realdocker-hostresources instead of leaking onto platform-managed runtimes. - Keep shared commercial upgrade navigation typed and destination-aware.
Shared paywall shells and upgrade actions must route internal billing or
cloud destinations through
frontend-modern/src/utils/upgradeNavigation.ts,frontend-modern/src/components/shared/UpgradeLink.tsx, andfrontend-modern/src/components/shared/useUpgradeNavigation.tsinstead of guessing from labels, hardcodingtarget="_blank", or callingwindow.open(...)from each feature surface. - Keep same-shell infrastructure route transitions on retained shared state.
frontend-modern/src/features/infrastructure/InfrastructurePageSurface.tsxmay show its full-page loading shell only before the first compatible resource snapshot exists; once a fresh canonical snapshot is already present in the shared app shell, top-level tab switches must reuse that state boundary instead of flashing a transient infrastructure page takeover between tabs.
Current State
SettingsTab no longer includes infrastructure-connections or
infrastructure-install. The single infrastructure-systems entry in
settingsNavCatalog.ts, settingsPanelRegistry.ts, and
settingsNavigationModel.ts replaces both. Panel routing within the
infrastructure area uses InfrastructurePanelStep in-page state.
Shared alert presentation surfaces (OverviewTab.tsx, HistoryTab.tsx,
AlertOverviewActiveAlertsSection.tsx, AlertHistoryTableSection.tsx,
AlertHistoryTableAlertRow.tsx, AlertOverviewAlertCard.tsx) no longer accept
hasAIAlertsFeature or runtimeCapabilitiesLoading props. Feature gating for
AI alerts flows through the shared entitlements layer; surfaces must not
re-introduce per-surface capability fetch props.
frontend-modern/src/components/Recovery/RecoverySummary.tsx gained an
aggregate health-state summary row. Per the Extension Points constraint, this
surface must continue to stay outside the shared hover-synchronization dialect;
new additions to RecoverySummary.tsx must not introduce row/group/chart hover
wiring without a separate governed product decision.
frontend-modern/src/components/Storage/useStorageSummaryCharts.ts now owns
the reusable polling/caching state for storage summary history, while
frontend-modern/src/features/storageBackups/storageCapacityDeltaPresentation.ts
keeps pool-growth label/tone formatting inside the shared feature presentation
layer. The storage page must keep reusing those shared owners instead of
rebuilding storage-history timers or byte-delta formatting inside row
components.
That same shared alerts feature boundary now also owns legacy shared-storage
override migration. frontend-modern/src/features/alerts/alertOverridesModel.ts
and frontend-modern/src/features/alerts/useAlertOverridesState.ts must
canonicalize per-node shared-storage override keys such as
Main-pve1-ceph-pool onto the current cluster-scoped storage resource id
before the thresholds table derives rows, so old Ceph override records survive
the v6 feature-shell path instead of silently disappearing from the live editor.
The frontend already has several guardrail tests. The next step is to keep
turning repeated local patterns into explicit shared primitives with hard usage
bounds, including provider-backed alert-history wording. frontend-modern/src/features/alerts/helpers.ts,
frontend-modern/src/features/alerts/tabs/HistoryTab.tsx, and
frontend-modern/src/features/alerts/OverviewTab.tsx must present VMware-
backed host and VM incidents with the shared resource-incident vocabulary
and existing alert-history shells instead of introducing VMware-only labels,
badges, or panel copy just because the underlying signal came from vSphere.
That same shared settings and modal boundary now also owns the public usage-data
vocabulary. frontend-modern/src/components/Settings/GeneralSettingsPanel.tsx,
frontend-modern/src/components/shared/whatsNewModalModel.ts,
frontend-modern/src/components/Settings/useSystemSettingsState.ts, and
frontend-modern/src/utils/systemSettingsPresentation.ts must present one
explicit Usage data and privacy model with the separate labels
Anonymous outbound telemetry and Disable local-only upgrade events, and
must describe telemetry with normalized release identity rather than falling
back to ambiguous telemetry, upgrade metrics, or raw-version wording.
Shared table, disclosure, and form primitives must also stay explicitly typed
at the browser edge. Summary rows may memoize repeated pending-update reads,
shared buttons must preserve discriminated disclosure props, toggle and a11y
helpers must expose exact event signatures, shared rows must accept typed
data-* props, and reporting-panel helpers must remain ES2020-safe instead of
depending on feature-local casts or newer string helpers.
That same shared settings-shell and banner boundary now also owns demo-mode
commercial suppression. frontend-modern/src/components/Settings/settingsNavCatalog.ts,
frontend-modern/src/components/Settings/settingsNavVisibility.ts,
frontend-modern/src/stores/sessionCapabilities.ts,
frontend-modern/src/stores/sessionPresentationPolicy.ts,
frontend-modern/src/stores/demoMode.ts,
frontend-modern/src/stores/license.ts,
frontend-modern/src/stores/licenseCommercial.ts,
frontend-modern/src/useAppRuntimeState.ts,
frontend-modern/src/components/shared/useTrialBannerState.ts, and
frontend-modern/src/components/shared/useMonitoredSystemLimitWarningBannerState.ts,
frontend-modern/src/components/shared/HistoryChartOverlay.tsx,
frontend-modern/src/features/patrol/PatrolIntelligenceBanners.tsx, and
frontend-modern/src/features/patrol/PatrolIntelligenceHeader.tsx
must consume one shared bootstrap truth from /api/security/status. The
backend capability fact sessionCapabilities.demoMode remains part of that
payload, but the browser-owned shared primitive is now the resolved
sessionPresentationPolicy contract, which hides billing tabs, trial nudges,
monitored-system warning banners, dashboard upsells, Patrol upgrade CTAs,
history-lock paywalls, and other public-demo commercial affordances when the
browser is rendering a public demo runtime.
That same shared settings-shell boundary also owns demo-mode organization
suppression. frontend-modern/src/components/Settings/settingsNavigationModel.ts,
frontend-modern/src/components/Settings/settingsNavCatalog.ts,
frontend-modern/src/components/Settings/settingsNavVisibility.ts,
frontend-modern/src/stores/sessionPresentationPolicy.ts, and
frontend-modern/src/useAppRuntimeState.ts must fail closed on organization
navigation and app-shell org chrome until the resolved presentation policy is
known, then keep org switchers, visible Default Organization labels, and
organization-scoped settings groups hidden when the browser is rendering a
public demo runtime.
Shared primitives must not perform their own ad hoc /api/health polling,
response-header inference, hostname heuristics, or per-banner demo branching;
the runtime bootstrap, shared presentation-policy store, and shared banner
hooks stay on one canonical owner so suppression stays coherent across
customer-facing surfaces.
That same shared primitive boundary now also treats runtime capability reads
and commercial reads as separate stores. Shared settings shells and banner
hooks may read feature truth from frontend-modern/src/stores/license.ts, but
commercial identity, upgrade routing, and trial state must stay in
frontend-modern/src/stores/licenseCommercial.ts, which suppresses public-demo
loads locally and defers its first fetch until the presentation policy has
resolved instead of depending on route-local guards.
That same shared primitive boundary now also centralizes authenticated-shell
commercial posture bootstrap. frontend-modern/src/useAppRuntimeState.ts
owns the first shared loadCommercialPosture() read after authenticated app
runtime has mounted, while frontend-modern/src/AppLayout.tsx,
frontend-modern/src/components/Settings/Settings.tsx, shared warning-banner
hooks, Patrol state hooks, and settings-panel state hooks must consume the
resolved store state instead of reissuing mount-time posture fetches from each
surface. Shared commercial posture loading may still dedupe or force-refresh
through the store for governed billing or first-run flows, but route-local or
panel-local bootstrap ownership is forbidden.
Storage disk drawers now also sit on that same shared-primitives floor.
frontend-modern/src/components/Storage/DiskDetail.tsx must render physical-
disk read, write, and busy charts through HistoryChart plus
useHistoryChartState, using the canonical physical-disk history resource id,
instead of reviving diskMetricsHistory, a page-local ring buffer, or another
storage-only live chart primitive for the same telemetry.
That same shared-primitive floor now also owns upgrade-navigation semantics.
frontend-modern/src/utils/upgradeNavigation.ts is the canonical typed
internal-vs-external destination helper, while
frontend-modern/src/components/shared/UpgradeLink.tsx and
frontend-modern/src/components/shared/useUpgradeNavigation.ts own how shared
paywall surfaces navigate those destinations. Feature shells may request a
commercial destination, but they must not re-decide whether that destination
opens in-app or in a new tab once the shared primitive exists.
That same shared-primitive floor now also owns prerelease shell guidance.
frontend-modern/src/AppLayout.tsx is the canonical authenticated-shell owner
for prerelease presentation, and the remaining user-facing treatment is the
compact Preview badge keyed from resolved release metadata. Feature pages,
settings panels, shared components, and route-local shells must not add a
second release-candidate banner, hardcoded GitHub release or feedback links,
or page-local prerelease notices once that shared shell contract exists.
Browser proof for that shell rule now lives in
tests/integration/tests/57-release-candidate-shell.spec.ts, which must keep
rc-channel builds banner-free while preserving the compact preview badge.
The subsystem registry now also requires explicit proof-policy coverage for all
shared runtime files, and shared-component guardrails fail if raw table
composition is reintroduced in new shared components outside the canonical
allowlist.
Retained-value query ownership is now part of that shared floor too.
frontend-modern/src/hooks/createNonSuspendingQuery.ts is the canonical
shared helper for page-local fetches that must stay inside the mounted
surface instead of falling through the app-level Loading view... fallback.
Feature slices such as recovery and infrastructure drawers may consume that
helper, but they must not fork new suspense-escape helpers once the shared
contract exists.
The settings reporting shell now also owns a deliberate split between
historical performance reports and current-state VM inventory export.
frontend-modern/src/components/Settings/ReportingPanel.tsx,
frontend-modern/src/components/Settings/useReportingPanelState.ts,
frontend-modern/src/components/Settings/reportingCatalogModel.ts,
frontend-modern/src/components/Settings/reportingPanelModel.ts, and
frontend-modern/src/components/Settings/reportingInventoryExportModel.ts must
keep those as separate operator jobs with separate request builders and success
copy, rather than collapsing inventory export back into the metrics-report
controls.
That same settings shell must now also render both historical performance
options and VM inventory schema from the backend-owned reporting catalog rather
than hardcoding panel copy, routes, or range presets in the frontend. The
frontend models may validate and present the catalog, but the canonical panel
title, descriptions, endpoints, filename prefixes, range windows, and column
list belong to the API reporting contract.
That same settings-shell boundary now also owns operator-facing docs referrals
for governed security panels. APIAccessPanel.tsx and
SecurityOverviewPanel.tsx must route scope and proxy-auth guidance through
the shared shipped-doc helper in frontend-modern/src/utils/docsLinks.ts
instead of hardcoding GitHub main URLs that can drift from the running
build, and tests/integration/tests/20-local-doc-links.spec.ts must keep
browser proof on those settings-shell surfaces.
That same settings-shell boundary now also owns the remediation framing for
Security Overview itself. SecurityOverviewPanel.tsx may not stop at a score
card and static best-practices copy once low-risk security debt has been
demoted out of the global banner; it must render explicit next-step hardening
actions on the canonical settings shell, source those actions from the shared
security presentation owner, and keep direct operator links pointed at the
owning auth, API-access, or shipped security-guide surface. The canonical
proof for that shell framing remains
frontend-modern/src/components/Settings/__tests__/settingsArchitecture.test.ts.
The same reporting catalog ownership now also governs the operator resource-
selection cap for performance reports. ReportingPanel.tsx and
ResourcePicker.tsx may present or enforce that limit, but they must receive
it from the backend-owned multiResourceMax definition rather than hardcoding
the reporting cap in frontend-local constants.
That same catalog-owned capability contract also governs which optional
performance-report controls appear at all. The settings shell and reporting
request builder may not assume metric filtering or custom titles are always
available; they must honor supportsMetricFilter and supportsCustomTitle
from the backend catalog and avoid emitting unsupported controls or request
parameters from frontend-local defaults.
That same backend-owned catalog also owns the initial reporting selections and
transport details. useReportingPanelState.ts,
reportingPanelModel.ts, and reportingInventoryExportModel.ts may not seed
format/range selections from legacy frontend constants or invent fallback
report endpoints, filename prefixes, default report titles, or range windows
or fallback filename date-stamp styles
when the catalog is
present; the first valid selection and all request semantics must come from the
parsed backend definition.
The same rule applies to VM inventory export transport details: request
builders and fallback filenames must derive the export format and extension
from the parsed inventory definition instead of hardcoding csv in frontend
helpers.
That same fallback contract also includes the single-report filename subject,
so frontend download builders may not substitute resource display names when
the catalog says fallback attachment names are keyed off canonical resource IDs.
That same reporting shell must also route failed catalog/report/export
responses through the shared API error extractor in frontend-modern/src/utils/apiClient.ts
rather than surfacing raw JSON payload text from response.text() directly in
warning UI.
That same reporting transport contract also means the frontend download path
must prefer the backend Content-Disposition filename over any locally built
fallback name when a report or inventory export response arrives.
That same settings shell must also read the reporting catalog for locked users,
not just entitled users, so the paywalled reporting panel does not drift onto a
separate frontend-owned title or description contract.
That same settings shell must now also treat the reporting catalog as the
feature-identity source once it loads. ReportingPanel.tsx and
useReportingPanelState.ts may use a generic loading or error shell before the
catalog is available, but they must not hardcode the reporting feature key or
the entitled and locked panel title and description once the catalog has
loaded.
The same metadata route is readable without the reporting feature gate, so the
settings shell must not delay the catalog fetch on licenseLoaded() before it
can render its canonical loading, locked, or entitled states.
That same shell must also stay usable against older Pulse backends that do not
yet expose /api/admin/reports/catalog. When that specific metadata route
returns 404, useReportingPanelState.ts may fall back to the governed legacy
performance-report transport (/api/reporting and /api/reporting/generate-multi)
so the reporting panel does not go dead on mixed-version installs, but that
compatibility path is intentionally report-only and must not invent the newer
catalog-owned VM inventory export surface.
ReportingPanel.tsx must therefore treat vmInventoryExport as optional when
it renders a governed reporting catalog. A legacy compatibility catalog with no
inventory export still owns a valid enabled reporting surface and must continue
to render the performance-report workflow instead of collapsing back to the
unavailable shell.
That same catalog load must also remain retryable after transient failure.
useReportingPanelState.ts may memoize or dedupe in-flight work, but it must
not permanently latch a failed first fetch and force operators to reload the
entire settings page before the reporting shell can recover.
That same catalog-owned contract also includes the locked teaser copy itself:
ReportingPanel.tsx may style or place the paywall content, but the locked
title and description must come from the parsed reporting catalog instead of
hardcoded component strings.
That same reporting catalog also owns the enabled-shell guidance callout that
explains when to use performance reports versus VM inventory export.
ReportingPanel.tsx may choose the presentation primitive, but the callout
title and description must come from the parsed catalog instead of a
frontend-local explainer paragraph.
The shared updates settings owner also defines the user-facing framing for
rc-tagged builds. frontend-modern/src/components/Settings/updatesSettingsModel.ts
and frontend-modern/src/utils/updatesPresentation.ts must present that
channel as a prerelease or preview path with manual validation expectations,
not as a near-ready release candidate promise.
The root app shell now also treats backend availability as distinct from
websocket liveness: frontend-modern/src/AppLayout.tsx and
frontend-modern/src/useAppRuntimeState.ts must keep the top-right connection
badge aligned to overall backend availability so a healthy dev/runtime backend
does not present the whole shell as reconnecting just because the live stream
is transiently renegotiating. That shell badge must now stay on an explicit
state model as well: healthy runtime, backend-healthy-but-stream-degraded,
full reconnect, and full disconnect are distinct operator states, and the
shared shell may not collapse them back into one generic reconnect label.
Shared feature presentation helpers under frontend-modern/src/features/ now
also need to preserve route-owned page-health semantics when the owning surface
is REST-backed: operators should only see reconnect or disconnected shells when
the route's own data contract is unhealthy, not because a global websocket
singleton is transiently reconnecting.
Those same feature-owned header badges must also stay aligned to the owning
runtime state instead of surfacing stale auxiliary counters as primary status;
an exhausted quickstart-credit badge cannot override an otherwise active Patrol
runtime unless quickstart exhaustion is the active blocker.
When those shared helpers surface quickstart state, the wording must stay
Patrol-scoped as well: the badge copy should talk about Patrol quickstart runs
or Patrol quickstart exhaustion on activated or trial-backed installs rather
than generic AI credits, anonymous free hosted AI, or broad hosted AI
availability.
The same shared shell rule applies to activation-gated availability copy: when
Patrol is blocked because quickstart lacks a server-verified install identity,
shared feature shells must preserve the backend activation-or-BYOK guidance
instead of translating that state into an exhausted-credit badge or a generic
hosted-AI upgrade message.
The same primitive boundary now also owns the first AI enable control in
AISettings.tsx: the primary toggle must remain explicitly addressable with a
stable accessible label, route through the runtime-backed quickstart
availability/blocked-reason state, and enable directly for quickstart-ready
installs instead of falling back to generic "first pressed toggle" selectors,
provider-model-load heuristics, or unconditional BYOK setup modals.
That same route-owned presentation rule also governs Patrol findings empty
states: shared section shells under frontend-modern/src/features/patrol/
must not render a green healthy empty state from 0 active findings alone
when the owning Patrol runtime or overall-health summary is degraded, blocked,
or not fully verified.
The same hierarchy also applies inside the Patrol summary shell: once the
primary summary card states Patrol's assessment and verification basis,
supporting metric strips under that card must stay metric-oriented and must
not repeat assessment or verification labels as a second compact verdict row.
That same summary shell should also keep the shared page-card base neutral:
severity belongs in compact header accents, icon chips, and badges rather
than turning the entire full-width summary into a tinted warning banner that
breaks the surrounding Pulse surface language.
That same summary-shell rule also applies to timing metadata: if the header,
verification card, or findings footer already presents the governed Patrol
activity timestamp, the summary chip row must not add another recency badge
that competes with those owned timing surfaces.
The same ownership split applies to supporting counts: if the Patrol summary
surface renders the metric strip for active findings, warnings, criticals, and
fixes, the primary summary card should not repeat those same counts in badge
form beside the assessment and verification copy.
That same ownership rule applies to empty-state timing metadata. When the
Patrol page header already carries schedule and recency context, the findings
empty state should not add its own footer for Last activity, Next run, or
run interval text.
Those supporting cards must also keep their content factual and count-based:
active findings, critical findings, warnings, and fixes are valid secondary
readouts, while labels such as Issues detected or Partial verification
belong only to the primary Patrol assessment and verification surfaces.
The same applies to Patrol operational context during active execution: the
shared feature surface may add an explicit run-in-progress badge, but any
activity support surface or integrated summary panel must remain factual
activity copy rather than shifting into a second Patrol verdict label while a
run is underway.
frontend-modern/src/components/shared/TagBadges.tsx is now also the
canonical tag-badge primitive. Dashboard workload rows and the unified-resource
detail drawer must import that shared owner instead of keeping a dashboard-local
tag badge variant or importing a feature-local path into infrastructure
surfaces.
That same owner now also holds the CSP-safe tag-dot rendering contract: tag
color and active-state emphasis must travel through SVG fill/stroke attributes
or stable classes, not inline background-color, box-shadow, or other
style= mutations that break the hosted demo CSP.
frontend-modern/src/components/Settings/OperationsPanel.tsx is now also the
canonical shared settings wrapper for operations-style panels such as
diagnostics, reporting, and system logs. Those surfaces must extend that owner
instead of rebuilding a local SettingsPanel wrapper, panel-header action
slot, or divided content-body framing inline.
The system logs operations surface now follows the same shell/runtime split as
the other modernized settings panels: frontend-modern/src/components/Settings/SystemLogsPanel.tsx
owns the operations framing and consumes the canonical stream-copy/status
helpers from frontend-modern/src/utils/systemLogsPresentation.ts, while
frontend-modern/src/components/Settings/useSystemLogsPanelState.ts owns the
stream lifecycle, buffering, level updates, and download action. Future system
logs work must extend that split instead of pulling EventSource, API calls,
notification flow, or customer-facing system-log copy back into the panel
render shell.
Shared trial CTA handling is now part of that same primitive boundary for
settings and shared paywalls. Shared/settings runtime owners must derive trial
eligibility and upgrade posture from the canonical commercial-posture
contract, including trial_eligible, while billing identity and plan detail
stay on the billing-only entitlements contract. Shared/settings runtime owners
must route operator-facing failure copy through
frontend-modern/src/utils/upgradePresentation.ts. The trial-start runtime
handoff itself is now centralized in
frontend-modern/src/utils/trialStartAction.ts; settings/shared paywalls and
onboarding surfaces must use that owner for redirect, success-notification, and
canonical denial handling instead of open-coding local startProTrial()
branches or re-interpreting backend status codes.
That same shared presentation owner also holds the canonical trial
rate-limit copy. When the shared commercial store provides
retryAfterSeconds, upgradePresentation.ts must render human guidance from
that canonical backoff instead of flattening the UI to a generic "try again
later" message; user-facing CTA surfaces must therefore inherit
Retry-After-backed copy through the shared helper path rather than
inventing lane-local 429 wording.
That same shared primitive boundary now also owns intent-level commercial
selectors for non-billing surfaces. Leaf/shared state such as
useTrialBannerState.ts,
useMonitoredSystemLimitWarningBannerState.ts, settings panel state, and
Patrol approval/header shells must consume selector helpers from
frontend-modern/src/stores/licenseCommercial.ts such as
canOfferCommercialTrial(), canStartCommercialTrial(),
isCommercialTrialActive(), commercialTrialDaysRemaining(), and
commercialOverflowDaysRemaining() instead of branching directly on raw
subscription_state, trial_eligible, or day-count fields.
That same owner also holds generic settings-paywall CTA labels. Runtime shells
such as AIRuntimeControlsSection.tsx and RelaySettingsPanel.tsx must source
shared labels like Upgrade to Pro and Start free trial from
upgradePresentation.ts rather than embedding local CTA strings in the panel
markup.
Top-level route files are now also expected to stay thin when a feature owns
the real product surface. frontend-modern/src/pages/Infrastructure.tsx now
acts only as the route boundary, while
frontend-modern/src/features/infrastructure/InfrastructurePageSurface.tsx
owns the shell, frontend-modern/src/features/infrastructure/useInfrastructurePageState.ts
owns page-control composition, persistence, and route composition,
frontend-modern/src/features/infrastructure/infrastructurePageModel.ts
owns filter/search/catalog derivation, and
frontend-modern/src/features/infrastructure/useInfrastructurePageRouteState.ts
owns infrastructure route/deep-link synchronization. Future feature
surfaces under frontend-modern/src/features/ should follow that same pattern
instead of letting page files accumulate route sync, filter, and modal
orchestration inline.
Infrastructure summary and detail surfaces now also use the shared normalized
identity lookup helper from frontend-modern/src/utils/resourceIdentity.ts
so dotted hostnames and alias variants stay consistent between the shared
table, drawer, and detail views instead of each component carrying its own
identifier-variant logic.
Those same surfaces also share the trimmed-string helper from
frontend-modern/src/utils/stringUtils.ts so shared components do not keep
their own copy of the same whitespace-trimming identity logic.
The shared infrastructure summary table now also follows the same
shell/runtime/model shape as the rest of the modernized primitives.
frontend-modern/src/components/shared/InfrastructureSummaryTable.tsx stays
the table shell, frontend-modern/src/components/shared/useInfrastructureSummaryTableState.ts
owns alert wiring, sort state, breakpoint state, and expanded-row lifecycle,
frontend-modern/src/components/shared/infrastructureSummaryTableModel.ts
owns sorting, count, identity-alias, and linked-agent derivation, and
frontend-modern/src/components/shared/InfrastructureSummaryTableRow.tsx
owns the per-row render/runtime surface. Future work should extend those
owners instead of pushing websocket, alert, or identity plumbing back into the
shared table shell.
The shared infrastructure selector now follows that same owner split.
frontend-modern/src/components/shared/InfrastructureSelector.tsx stays the
render shell, frontend-modern/src/components/shared/useInfrastructureSelectorState.ts
owns selected-node state, tab-reset and escape-key lifecycle, plus hook-backed
resource and recovery composition, and
frontend-modern/src/components/shared/infrastructureSelectorModel.ts owns
resource-family counts, agent-backed node-summary projection, unified-node and
PBS-instance projection, and recovery backup-count derivation. Future
infrastructure-selector work should extend those owners instead of pushing
resource aggregation or selection lifecycle back into the shared shell.
That shared selector projection must also preserve canonical local operator
identity for agent-backed infrastructure labels. Governed or AI-safe resource
summaries may inform policy/detail surfaces, but the selector's summary and
drawer-facing agent labels must continue to use the same local instance
identity boundary as the operator-facing infrastructure tables so multiple PBS,
PMG, or other governed resources remain distinguishable.
The shared infrastructure details drawer now follows that same owner split.
frontend-modern/src/components/shared/InfrastructureDetailsDrawer.tsx stays
the render shell, frontend-modern/src/components/shared/useInfrastructureDetailsDrawerState.ts
owns tab-selection runtime, and
frontend-modern/src/components/shared/infrastructureDetailsDrawerModel.ts
owns canonical metadata-id and discovery-hostname derivation. Future
infrastructure-details-drawer work should extend those owners instead of
pushing tab state or resource-identity normalization back into the shared
shell.
The shared interactive sparkline now follows that same split.
frontend-modern/src/components/shared/InteractiveSparkline.tsx stays the
render shell, frontend-modern/src/components/shared/useInteractiveSparklineState.ts
owns hover state, RAF throttling, canvas draw scheduling, and resize lifecycle,
and frontend-modern/src/components/shared/interactiveSparklineModel.ts owns
sparkline downsampling, gap segmentation, axis-tick math, and hover-selection
policy. Future sparkline work should extend those owners instead of pushing
canvas scheduling or chart-shape math back into the shared component shell.
That same sparkline boundary now also owns floating tooltip shell routing:
local hover tooltips must derive viewport anchor coordinates from the shared
runtime/model path and render through
frontend-modern/src/components/shared/TooltipPortal.tsx, not as HTML
foreignObject shells inside the preserveAspectRatio="none" chart SVG where
cross-browser scaling can stretch the tooltip surface or drop its semantic
shell styling.
That same shared sparkline boundary now also owns active-series isolation
metadata. The shell may expose data-active-series-display and
data-rendered-series-count for proof and inspection, but only the shared
runtime/model owners may decide whether a hovered or focused series is merely
emphasized or fully isolated; feature shells must not fork their own row-hover
line filtering.
The dashboard overview trend cards now also have an explicit shared-shell
obligation: frontend-modern/src/features/dashboardOverview/TrendCharts.tsx
must treat missing infrastructure history as a governed empty state rather than
as a silent blank sparkline box. Error copy and empty-history copy belong to
the feature shell, while the data path and chart-shaping logic must stay in the
owned hook/model layers that feed it.
That same dashboard overview shell also owns workload discoverability copy on
the KPI strip. frontend-modern/src/features/dashboardOverview/KPIStrip.tsx
and frontend-modern/src/utils/dashboardKpiPresentation.ts must keep the
Workloads card explicit that the unified workloads surface contains VMs,
containers, and pods, rather than assuming operators will infer that mapping
from the route label alone.
That same dashboard boundary now also owns the shared dashboard presentation
helpers through frontend-modern/src/utils/dashboardEmptyStatePresentation.ts,
frontend-modern/src/utils/dashboardGuestPresentation.ts,
frontend-modern/src/utils/dashboardKpiPresentation.ts,
frontend-modern/src/utils/dashboardMetricPresentation.ts, and
frontend-modern/src/utils/dashboardTrendPresentation.ts. Dashboard loading,
disconnect, and empty states; guest backup/disk fallback copy; KPI card
framing; status-badge, delta, percent, and action-priority formatting; and
trend palette/error copy must extend those helpers instead of being redefined
inline in route shells, guest rows, or overview cards.
That shell must also stay passive with respect to data ownership: dashboard
trend cards may render the summary-range controls and operator-facing empty or
error copy, but they must not reintroduce route-local metrics-history fetch
loops for CPU and memory sparklines when the canonical infrastructure summary
surface already owns the chart contract.
The shared density map now follows that same owner split.
frontend-modern/src/components/shared/DensityMap.tsx stays the render shell,
frontend-modern/src/components/shared/useDensityMapState.ts owns hover
signals, canvas draw lifecycle, and resize handling, and
frontend-modern/src/components/shared/densityMapModel.ts owns bucket/window
math, hover target selection, focused-series tooltip detail, and density-cell
opacity rules. Future density-map work should extend those owners instead of
pushing canvas lifecycle, tooltip shaping, or chart math back into the shared
shell.
The shared trial banner now follows that same owner split.
frontend-modern/src/components/shared/TrialBanner.tsx stays the render
shell, frontend-modern/src/components/shared/useTrialBannerState.ts owns
entitlement load, snooze lifecycle, and upgrade-link runtime, and
frontend-modern/src/components/shared/trialBannerModel.ts owns day-count
normalization, tone policy, and display labels. Future trial-banner work
should extend those owners instead of pushing entitlement orchestration,
snooze state, or tone math back into the shared shell.
The shared column picker now follows that same owner split.
frontend-modern/src/components/shared/ColumnPicker.tsx stays the render
shell, frontend-modern/src/components/shared/useColumnPickerState.ts owns
dropdown open state and outside-click listener lifecycle, and
frontend-modern/src/components/shared/columnPickerModel.ts owns hidden-column
count, reset visibility policy, and column-option text-class/copy policy.
Future column-picker work should extend those owners instead of pushing
document-level listener logic or column-count policy back into the shell.
The shared tag input now follows that same owner split.
frontend-modern/src/components/shared/TagInput.tsx stays the render shell,
frontend-modern/src/components/shared/useTagInputState.ts owns input state,
container-focus runtime, and tag add/remove/backspace orchestration, and
frontend-modern/src/components/shared/tagInputModel.ts owns delimiter keys,
placeholder policy, remove-title copy, and canonical next-tag derivation.
Future tag-input work should extend those owners instead of pushing DOM reach-in
or tag-mutation policy back into the shell.
The shared scroll-to-top button now follows that same owner split.
frontend-modern/src/components/shared/ScrollToTopButton.tsx stays the render
shell, frontend-modern/src/components/shared/useScrollToTopButtonState.ts
owns scroll-listener lifecycle, visible state, and smooth-scroll runtime, and
frontend-modern/src/components/shared/scrollToTopButtonModel.ts owns
scrollable-ancestor discovery, visibility threshold policy, aria label, and
button class policy. Future scroll-to-top work should extend those owners
instead of pushing scroll-container discovery or listener lifecycle back into
the shell.
The shared toggle now follows that same owner split.
frontend-modern/src/components/shared/Toggle.tsx stays the render shell,
frontend-modern/src/components/shared/useToggleState.ts owns disabled gating
plus the synthetic toggle change-event runtime, and
frontend-modern/src/components/shared/toggleModel.ts owns size resolution,
track/knob/container class policy, and the canonical toggle event type.
Future toggle work should extend those owners instead of pushing synthetic
event behavior or size/class policy back into the shell.
The shared status badge now follows that same owner split.
frontend-modern/src/components/shared/StatusBadge.tsx stays the render shell,
frontend-modern/src/components/shared/useStatusBadgeState.ts owns disabled
gating and click runtime, and
frontend-modern/src/components/shared/statusBadgeModel.ts owns size padding,
label/title fallback policy, and status-badge class selection. Future status
badge work should extend those owners instead of pushing label/title policy or
disabled click handling back into the shell.
The shared segmented selector now follows that same owner split.
frontend-modern/src/components/shared/FilterButtonGroup.tsx stays the render
shell, frontend-modern/src/components/shared/useFilterButtonGroupState.ts
owns variant resolution plus disabled selection/change runtime, and
frontend-modern/src/components/shared/filterButtonGroupModel.ts owns the
variant class catalog, compact-label policy, and segmented button class
selection. Future filter-button-group work should extend those owners instead
of pushing label truncation or segmented variant policy back into the shell.
The shared selection-card primitive now follows that same owner split.
frontend-modern/src/components/shared/SelectionCardGroup.tsx stays the render
shell, frontend-modern/src/components/shared/useSelectionCardGroupState.ts
owns variant resolution plus disabled selection/change runtime, and
frontend-modern/src/components/shared/selectionCardGroupModel.ts owns the
tone fallback, group/button class catalog, and title/description presentation
policy. Future selection-card-group work should extend those owners instead of
pushing tone or active-card presentation logic back into the shell.
The shared dialog now follows that same owner split.
frontend-modern/src/components/shared/Dialog.tsx stays the render shell,
frontend-modern/src/components/shared/useDialogState.ts owns focus trap,
body-scroll lock, previous-focus restoration, shared blocking-dialog
visibility, and backdrop-close runtime, and
frontend-modern/src/components/shared/dialogModel.ts owns focusable-element
lookup plus layout and panel class policy. Future dialog work should extend
those owners instead of pushing focus-trap lifecycle or layout policy back into
the shared shell.
App-shell consumers such as frontend-modern/src/App.tsx and
frontend-modern/src/AppLayout.tsx may read that shared blocking-dialog state
to suppress background affordances, but they must not reimplement their own
parallel modal-stack bookkeeping.
The shared history chart now follows the same owner shape.
frontend-modern/src/components/shared/HistoryChart.tsx stays the render
shell, frontend-modern/src/components/shared/useHistoryChartState.ts owns
license gating, trial actions, history fetch/refresh, canvas draw lifecycle,
and hover state, and frontend-modern/src/components/shared/historyChartModel.ts
owns tooltip formatting, scale and axis math, and closest-point selection.
Future history-chart work should extend those owners instead of pushing fetch,
license, or canvas math back into the shared component shell.
The remaining header, overlay, and tooltip render surfaces now live in
frontend-modern/src/components/shared/HistoryChartHeader.tsx,
frontend-modern/src/components/shared/HistoryChartOverlay.tsx, and
frontend-modern/src/components/shared/HistoryChartTooltip.tsx instead of
re-accumulating those sections inline in the shell.
That tooltip owner now also holds the CSP-safe hover contract: chart tooltips
must render inside the chart surface with model-owned layout and SVG/attribute
positioning, not through fixed portals or inline left/top style attributes
that violate the public demo CSP.
The shared container update badge now follows that same owner split.
frontend-modern/src/components/shared/ContainerUpdateBadge.tsx stays the
render surface for the badge, icon, and update button shells,
frontend-modern/src/components/shared/useContainerUpdateButtonState.ts owns
Docker update mutation flow, persistent update-store state, settings gating,
and button lifecycle, and
frontend-modern/src/components/shared/containerUpdateBadgeModel.ts owns badge
and button tooltip formatting, class selection, and label/state presentation.
Future container-update work should extend those owners instead of pushing
store wiring, settings reads, or mutation flow back into the shared shell.
The shared web interface URL field now follows that same owner split.
frontend-modern/src/components/shared/WebInterfaceUrlField.tsx stays the
render shell, frontend-modern/src/components/shared/useWebInterfaceUrlFieldState.ts
owns metadata fetch/save/remove lifecycle, success/error state, and suggested
URL runtime, and frontend-modern/src/components/shared/webInterfaceUrlFieldModel.ts
owns URL validation, target-label normalization, and suggested-URL presentation
rules. The shared primitive now also supports an embedded mode with a caller-
owned title so feature drawers can place web-interface controls inside a larger
access surface without forking the save/remove/runtime behavior. Future
web-interface URL work should extend those owners instead of pushing metadata
transport or validation back into the shared shell.
The shared help icon now follows that same owner split.
frontend-modern/src/components/shared/HelpIcon.tsx stays the render shell,
frontend-modern/src/components/shared/useHelpIconState.ts owns open state,
popover-position lifecycle, and global click/escape listeners, and
frontend-modern/src/components/shared/helpIconModel.ts owns help-content
resolution, icon sizing, missing-content warnings, and popover-position math.
Future help-icon work should extend those owners instead of pushing registry
lookups or DOM listener lifecycle back into the shared shell.
The shared mobile nav now follows that same owner split.
frontend-modern/src/components/shared/MobileNavBar.tsx stays the render
shell, frontend-modern/src/components/shared/useMobileNavBarState.ts owns
fade signals, scroll and resize listeners, active-tab centering, and click
handoff lifecycle, and
frontend-modern/src/components/shared/mobileNavBarModel.ts owns platform and
utility tab ordering, alert badge counts, fade-state derivation, and tab
button class policy. Future mobile-nav work should extend those owners instead
of pushing tab-order or DOM lifecycle logic back into the shared shell. With
support/admin tools moved under Settings, that utility ordering must no longer
reserve a standalone operations slot; alerts, Patrol, and Settings are the
remaining authenticated utility tabs.
The shared command palette now follows that same owner split.
frontend-modern/src/components/shared/CommandPaletteModal.tsx stays the
render shell, frontend-modern/src/components/shared/useCommandPaletteState.ts
owns query state, open-reset/focus lifecycle, route-path wiring, and Enter-key
selection, and frontend-modern/src/components/shared/commandPaletteModel.ts
owns canonical command construction plus query normalization and filtering
policy. Future command-palette work should extend those owners instead of
pushing route construction or search policy back into the shared shell.
The shared search field now follows that same owner split.
frontend-modern/src/components/shared/SearchField.tsx stays the render shell,
frontend-modern/src/components/shared/useSearchFieldState.ts owns focused-
Escape clear/blur behavior and input-ref lifecycle, and
frontend-modern/src/components/shared/searchFieldModel.ts owns clear/shortcut
visibility rules plus trailing-control padding policy. Future search-field work
should extend those owners instead of pushing event behavior or layout policy
back into the shared shell.
The shared search input now follows that same owner split.
frontend-modern/src/components/shared/SearchInput.tsx stays the render shell,
frontend-modern/src/components/shared/useSearchInputState.ts owns input-ref
lifecycle, type-to-search registration, and enhancement runtime composition,
and frontend-modern/src/components/shared/searchInputModel.ts owns the shared
search-input contract plus shortcut-hint and trailing-control policy. Future
search-input work should extend those owners instead of pushing type-to-search
or enhancement wiring back into the shared shell.
The shared page-controls bar now follows that same owner split.
frontend-modern/src/components/shared/PageControls.tsx stays the render shell
for canonical page-level control composition, while
frontend-modern/src/components/shared/FilterToolbar.tsx owns the shared
search-row, filter-row, and inline-leading-slot layout surface. Monitoring
pages that need workspace tabs or count chips next to search should route that
through the shared searchLeading slot instead of recreating a second local
header strip above the control bar.
That same shared filter-toolbar boundary also owns controlled select continuity
when filter options materialize asynchronously. LabeledFilterSelect must keep
the caller-owned value visibly selected after option children arrive so
dashboard, recovery, and other canonical filter bars do not drop their active
selection until the operator reopens the control.
That same boundary also owns live option propagation through shared page-control
composition. Callers such as storage and recovery must pass source/filter
option collections through reactive accessors instead of snapshot arrays when
those options depend on post-load unified-resource state, so the shared toolbar
can reconcile late-arriving options and preserved route selections without
requiring page-local reset hacks.
When those workspace tabs need an embedded control-bar treatment, they should
still stay on the one canonical frontend-modern/src/components/shared/Subtabs.tsx
primitive and reuse the established shell, list, and button class pattern
already proven on owning surfaces like operations rather than introducing new
variant APIs on the primitive.
The search-input enhancement surfaces now follow that same owner split.
frontend-modern/src/components/shared/SearchInputEnhancements.tsx stays the
render shell, frontend-modern/src/components/shared/useSearchInputEnhancements.ts
owns search-history persistence, menu-open lifecycle, blur commit policy, and
tips/history interaction runtime, and
frontend-modern/src/components/shared/searchInputEnhancementsModel.ts owns
history-toggle copy plus history-menu button and row class policy. Future
search-input-enhancement work should extend those owners instead of pushing
history copy or menu presentation policy back into the shell.
The shared search tips popover now follows that same owner split.
frontend-modern/src/components/shared/SearchTipsPopover.tsx stays the render
shell, frontend-modern/src/components/shared/useSearchTipsPopoverState.ts
owns open-state, pointer/focus continuity, and outside-click/Escape listener
runtime, and frontend-modern/src/components/shared/searchTipsPopoverModel.ts
owns trigger variant, label/id defaults, hover policy, and trigger/popover
class selection. Future search-tips work should extend those owners instead of
pushing listener lifecycle or trigger policy back into the shared shell.
The shared what's-new modal now follows that same owner split.
frontend-modern/src/components/shared/WhatsNewModal.tsx stays the render
shell, frontend-modern/src/components/shared/useWhatsNewModalState.ts owns
local-storage dismissal, session dismissal, step progression, spotlight target
resolution, direct stop selection, and overlay placement/runtime behavior, and
frontend-modern/src/components/shared/whatsNewModalModel.ts owns the feature
tour catalog, telemetry copy, labels, and canonical docs/privacy links. Future
what's-new work should extend those owners instead of pushing dismissal state,
spotlight runtime, product copy, or external links back into the shared shell.
The v6 welcome surface is one guided spotlight tour, not a modal plus a second
dashboard-only migration hint: it must dim the live app, glow the real
primary-navigation target being described, and keep route-orientation copy on
the existing welcome flow instead of layering a duplicate in-product banner.
Its primary job is fast route orientation. The modal should explain what each
top-level area is for in plain product language, so operators can understand
the new navigation in one pass without needing historical layout context.
That copy should stay direct and present-tense. Each guided step should say
what the destination does, not depend on v5 comparisons, migration framing, or
older information architecture to make sense.
That guided welcome surface should stay compact. The canonical shape is a
coachmark-sized card centered on the current destination with one short
step-specific sentence, a small clickable step strip, and minimal footer
controls. It must not grow back into a large sectioned explainer when one
sentence would do the job.
The guided stop map inside that welcome surface is interactive, not decorative:
operators must be able to jump directly to any tour step from the stop list,
and desktop layouts may widen the panel enough to keep step labels readable
without overlapping or collapsing into clipped pills. That map should read as
numbered wayfinding, not placeholder onboarding chrome: concise numeric badges
plus section titles are preferred over repeated Stop N copy or other filler
labels that add noise without helping orientation.
That same welcome surface must stay inside Pulse's existing flat visual
language. The shell, step map, telemetry note, and supporting actions should
use bordered flat fills and normal app radii instead of gradient washes,
glassmorphism, or other marketing-style promo chrome that drifts from the rest
of the product.
Secondary disclosures such as telemetry must stay subordinate to that
orientation job: keep them as footer-level links into the canonical
privacy/settings surfaces, and do not let them crowd out the migration
wayfinding copy. The supporting docs CTA on that surface should likewise stay
route-oriented: use a neutral Navigation guide label and plain present-tense
copy that helps operators understand the current IA, rather than reviving
Migration guide branding that pulls the tour back into v5 historical framing.
That state owner now also owns public-demo suppression: the modal must stay
closed until sessionPresentationPolicyResolved() is true and must fail closed
when presentationPolicyIsDemoMode() resolves true, so the public demo does
not front-load product migration onboarding ahead of the actual surface.
Canonical customer disclosures inside those shared shells now route through
frontend-modern/src/utils/docsLinks.ts, so settings and what's-new privacy
links resolve to shipped /docs/... assets instead of hard-coded GitHub
main URLs that can drift from the running build.
The shared summary strip primitives now follow that same owner split.
frontend-modern/src/components/shared/SummaryPanel.tsx and
frontend-modern/src/components/shared/SummaryMetricCard.tsx stay the render
shells for summary-frame spacing and card density, while monitoring surfaces
such as recovery, infrastructure, workloads, and storage only choose from the
owned shared density modes instead of forking summary spacing with feature-
local padding hacks. Future summary-density work should extend those shared
primitives rather than hard-coding compact card chrome inside one surface.
The shared tooltip now follows that same owner split.
frontend-modern/src/components/shared/Tooltip.tsx stays the render shell and
singleton API boundary, frontend-modern/src/components/shared/useTooltipState.ts
owns tooltip positioning lifecycle, RAF scheduling, and singleton visibility
state, and frontend-modern/src/components/shared/tooltipModel.ts owns tooltip
sanitization plus viewport-clamped positioning math. Future tooltip work should
extend those owners instead of pushing singleton state, DOM measurement, or
sanitization logic back into the shared shell. Shared portal-mounted tooltip
shells such as frontend-modern/src/components/shared/TooltipPortal.tsx must
use the same semantic surface tokens as the canonical tooltip instead of
introducing light-mode-inverted palettes.
That same tooltip owner now also holds the CSP-safe portal contract: shared
tooltip shells must render through SVG/attribute positioning and viewport-
clamped layout helpers rather than fixed inline left/top style attributes.
When a shared portal tooltip is already visible, that same owner must
reschedule positioning on live coordinate and viewport changes so chart hover
tooltips keep following the active pointer instead of sticking to their first
anchor.
The shared collapsible search input now follows that same owner split.
frontend-modern/src/components/shared/CollapsibleSearchInput.tsx stays the
render shell, frontend-modern/src/components/shared/useCollapsibleSearchInputState.ts
owns expand/collapse state, focus choreography, and type-to-search handoff, and
frontend-modern/src/components/shared/collapsibleSearchInputModel.ts owns
trigger-label, expanded-visibility, and full-width layout policy. Future
collapsible-search work should extend those owners instead of pushing
expand/collapse runtime or layout rules back into the shared shell.
The shared pulse data grid now follows that same owner split.
frontend-modern/src/components/shared/PulseDataGrid.tsx stays the render
shell, frontend-modern/src/components/shared/usePulseDataGridState.ts owns
breakpoint-driven min-width selection and stable-row reconciliation, and
frontend-modern/src/components/shared/pulseDataGridModel.ts owns alignment
class policy plus interactive-target row-click protection. Future pulse-data-
grid work should extend those owners instead of pushing breakpoint lifecycle or
interaction policy back into the shared shell.
The audit log settings surface now follows that same owner split.
frontend-modern/src/components/Settings/AuditLogPanel.tsx stays the canonical
SettingsPanel shell, while
frontend-modern/src/components/Settings/useAuditLogPanelState.ts owns the
license/paywall lifecycle, persisted filters, verification flow, and audit-log
fetch orchestration. The shell must not re-accumulate localStorage or API
runtime logic inline.
The audit webhook settings surface now follows that same owner split.
frontend-modern/src/components/Settings/AuditWebhookPanel.tsx stays the
canonical SettingsPanel shell, while
frontend-modern/src/components/Settings/useAuditWebhookPanelState.ts owns the
license/paywall lifecycle, webhook fetch/save flow, validation, and trial
startup orchestration. The shell must not re-accumulate API calls or paywall
tracking inline.
The diagnostics settings surface now follows that same owner split.
frontend-modern/src/components/Settings/DiagnosticsPanel.tsx stays the
top-level diagnostics shell, while
frontend-modern/src/components/Settings/useDiagnosticsPanelState.ts,
frontend-modern/src/components/Settings/DiagnosticsResultsPanel.tsx,
frontend-modern/src/components/Settings/diagnosticsModel.ts, and
frontend-modern/src/utils/diagnosticsPresentation.ts own the diagnostics
run/export lifecycle, results rendering, sanitization/model helpers, and
customer-facing diagnostics copy. The shell must not re-accumulate inline API
calls, export-download plumbing, diagnostics-card composition, or diagnostics
surface copy.
That same diagnostics owner split now also covers local commercial funnel
rendering: if diagnostics surfaces expose self-hosted pricing, checkout, or
activation summaries, DiagnosticsResultsPanel.tsx and diagnosticsModel.ts
must own the card composition, label humanization, and typed payload shape,
while the shell remains a layout/composition owner and does not reintroduce
inline diagnostics fetches or commerce-specific rendering logic.
The settings shell registry now also treats extracted feature prop contracts as
canonical shell inputs instead of reaching back into feature panels for type
ownership. frontend-modern/src/components/Settings/useSettingsPanelRegistry.tsx
must consume the direct Proxmox panel contract through
frontend-modern/src/components/Settings/proxmoxSettingsModel.ts, so the
registry stays a shell/composition owner and does not depend on
ProxmoxSettingsPanel.tsx as though the panel still owned the runtime model.
The retired /operations route is now a thin compatibility redirect only.
frontend-modern/src/pages/Operations.tsx may normalize legacy /operations/*
links into the canonical Settings support routes, but diagnostics, reports,
and logs now belong to the shared Settings shell instead of a bespoke page-
local tab surface. Support-only navigation must therefore route through the
shared settings owners rather than rebuilding a second route-level shell, and
public demo posture must keep those support entries hidden from the Settings
navigation instead of reviving a standalone operations page.
that are unavailable in demo mode.
The dashboard overview route now follows that same feature-owner pattern for
its dashboard-specific summary surfaces. frontend-modern/src/pages/Dashboard.tsx
stays the route shell, while frontend-modern/src/features/dashboardOverview/
owns the dashboard-specific action, KPI, problem-resource, trend, and
customization surfaces. Lane-owned widgets like recent alerts, storage,
and recovery must continue to route through their own subsystem owners instead
of drifting back into a page-local dashboard panel cluster.
That same dashboard overview boundary owns the first-viewport estate
orientation contract for the v6 landing page. EstateSummaryPanel.tsx and
estateSummaryModel.ts in frontend-modern/src/features/dashboardOverview/
must derive system count, health, source coverage, and freshness from the
canonical connected-infrastructure projection, fall back only to the compact
dashboard summary that the route already owns, and keep the explicit
Infrastructure handoff above detailed problem, storage, recovery, or trend
rows without restoring platform-special navigation.
That first-viewport copy must distinguish system-level estate health from
resource, alert, storage, or recovery issues that remain elsewhere on the
dashboard, and partial/empty dashboard states must describe synchronization or
infrastructure-source onboarding in operator terms instead of exposing
implementation fallback language.
The recovery feature shell now also depends on the shared
frontend-modern/src/components/shared/Subtabs.tsx primitive for its primary
protected-items versus recovery-events workspace switch. The recovery lane may
own the active view and route-state semantics, but the top-level tab framing
must stay on the canonical shared subtabs control instead of reviving a
recovery-local switcher pattern. When recovery embeds that switcher inside the
page shell, it should follow the same ordering already used by storage: shared
subtabs row first, shared controls card second, and data card after that. The
contained styling should come from the same canonical subtabs shell, list, and
button class treatment already used by established Pulse surfaces rather than
from a recovery-only variant boundary, adjacent chip row, or recovery-local
filter-row embedding.
The shared table primitives now also need to preserve caller-owned separator
styling. TableHeader and TableBody may provide canonical default borders
and dividers, but when a caller supplies explicit border or divide classes the
shared primitive must defer to that local contract instead of silently forcing
the default separator treatment back into the rendered DOM.
That same shared table boundary now owns CSP-safe sizing for infrastructure
tables and metric bars. frontend-modern/src/components/Infrastructure/useUnifiedResourceTableState.ts
and frontend-modern/src/components/Infrastructure/unifiedResourceTableStateModel.ts
must express table layout and column sizing as shared class/attribute
presentation instead of inline style= maps, and
frontend-modern/src/components/shared/ProgressBar.tsx must render fill width
through DOM attributes rather than inline width styles. Infrastructure host and
service tables may still vary by breakpoint and column family, but they must do
so through the shared presentation owner instead of lane-local style objects
that break the public demo CSP.
That same shared-boundary rule applies to summary density. The shared compact
mode on SummaryPanel.tsx and SummaryMetricCard.tsx exists for genuinely
dense monitoring surfaces, but pages that are trying to align with the normal
Pulse monitoring scan path should stay on the default shared density instead of
using page-local compact overrides by habit.
That same recovery shell boundary now also owns one canonical top-level filter
controller in
frontend-modern/src/features/recovery/useRecoverySurfaceState.ts. Route-backed
recovery filters such as the provider-neutral itemType selector must be
derived, normalized, and fanned out to inventory, history, activity, facets,
and series consumers from that shared state owner rather than being recreated
as page-local toolbar state inside individual recovery sections.
That same shared recovery filter boundary also owns canonical recovery
item-type derivation through
frontend-modern/src/utils/recoveryItemTypePresentation.ts. Recovery shell
state, tables, summaries, and point-detail surfaces must resolve rollup and
point item types through the shared presenter helpers instead of repeating
display.itemType / subjectType / subjectRef.type fallback chains in
page-local consumers.
That same shared recovery decode boundary also owns canonical recovery display
shape. frontend-modern/src/utils/recoveryPlatformModel.ts,
frontend-modern/src/hooks/useRecoveryPoints.ts, and
frontend-modern/src/hooks/useRecoveryRollups.ts must normalize legacy
transport display aliases like subjectLabel and subjectType into canonical
runtime itemLabel and itemType fields before recovery presenters consume
the model.
The same shared recovery-column boundary must keep legacy subject and
source column ids at migration-only scope once
frontend-modern/src/hooks/useColumnVisibility.ts owns alias rewrites.
Recovery table runtime helpers and render switches should operate on canonical
item and platform ids rather than carrying the deleted ids as live cases.
That same shared recovery state owner now also keeps platform as the
canonical route and transport filter name for operator-facing recovery links,
while any accepted legacy provider aliases remain parser compatibility only.
Caller-facing shared recovery route builders must therefore stay
platform-first as well: compatibility provider aliases may be accepted while
parsing legacy links, but they should not remain a first-class input on new
recovery link construction helpers.
Recovery frontend decode and derived option builders must treat payload
platform / platforms as the canonical response fields and only fall back
to legacy provider / providers aliases for compatibility, so route,
filter, and table state do not keep backend-era vocabulary alive as the
default client model.
That normalization belongs at the shared recovery transport boundary in
frontend-modern/src/hooks/useRecoveryPoints.ts and
frontend-modern/src/hooks/useRecoveryRollups.ts, not in individual tables,
drawers, or summary cards. Recovery components should receive canonical
platform-first runtime models rather than re-deriving legacy alias fallback
locally.
Recovery section owners under frontend-modern/src/components/Recovery/ must
consume that shared platform filter surface directly. They must not keep
recovery-local provider route/query vocabulary alive behind renamed labels,
or the UI will drift back to backend-shaped navigation even when the copy says
Platform.
That same shared recovery filter owner must also preserve route-owned platform
visibility while transport-backed options are still hydrating. If
frontend-modern/src/features/recovery/useRecoverySurfaceState.ts restores a
canonical platform selection such as truenas from the route before the
rollups, points, or facets payloads arrive, it must keep that selected
platform present in the option set so the shared LabeledFilterSelect shows
the owned value immediately instead of flashing back to All Platforms until
recovery data warms.
frontend-modern/src/utils/problemResourcePresentation.ts now also belongs to
that same dashboard overview boundary so the problem-resource severity contract
stays shared with ProblemResourcesTable.tsx instead of floating as an
unowned helper.
That same dashboard overview boundary must consume the governed Patrol finding
presentation helpers when it surfaces Patrol findings in compact form. In
frontend-modern/src/features/dashboardOverview/ActionRequiredPanel.tsx,
Patrol-owned runtime findings must use the shared compact badge, title, and
primary-action/manual-control contracts from
frontend-modern/src/utils/aiFindingPresentation.ts rather than rendering raw
Pulse Patrol: titles or generic snooze/dismiss controls that the Patrol
runtime lifecycle rejects.
That same boundary must also consume the shared attention-queue ordering from
frontend-modern/src/utils/aiFindingPresentation.ts through
frontend-modern/src/hooks/useDashboardActions.ts, so Patrol-blinding runtime
issues sort ahead of same-severity infrastructure findings in the dashboard
action panel instead of inheriting arbitrary store order.
Feature-owned alert shells under frontend-modern/src/features/alerts/ now
also treat shared action runtime as a first-class feature owner instead of
rebuilding it per surface. The overview shell and dashboard recent-alerts panel
must both compose
frontend-modern/src/features/alerts/useAlertAcknowledgementState.ts for
acknowledge/restore behavior rather than keeping duplicate API and notification
logic inline in useAlertOverviewState.ts or
frontend-modern/src/components/Alerts/RecentAlertsPanel.tsx.
The same feature-owner rule now applies to the alert scheduling surface:
frontend-modern/src/features/alerts/tabs/ScheduleTab.tsx must remain the
schedule render shell, while
frontend-modern/src/features/alerts/useAlertScheduleState.ts owns schedule
reset/update policy and canonical default application. The tab should not
re-accumulate quiet-hours, cooldown, grouping, or escalation mutation logic
inline.
The thresholds editor now follows that same split more tightly:
frontend-modern/src/features/alerts/thresholds/hooks/useThresholdsTableState.ts
must stay the table-shell owner for route sync and local UI state, while
frontend-modern/src/features/alerts/thresholds/hooks/useThresholdsData.ts
stays the composition shell for threshold resource-family projectors,
frontend-modern/src/features/alerts/thresholds/hooks/useThresholdsRecoveryDefaultsState.ts
owns backup/snapshot default sanitization and factory-drift policy, and
frontend-modern/src/features/alerts/thresholds/thresholdsOverrideMutationModel.ts
owns pure override upsert/hysteresis/state-strip helpers,
frontend-modern/src/features/alerts/thresholds/hooks/useThresholdsOverrideMutations.ts
owns threshold-save and backup/snapshot override persistence, and
frontend-modern/src/features/alerts/thresholds/hooks/useThresholdsAvailabilityMutations.ts
owns availability-state policy and alert-removal side effects. The table-shell
hook should not re-accumulate raw override mutation logic,
recovery-threshold defaults policy, or resource-family projection engines
inline.
The updates settings surface now follows the same presentation-owner rule.
frontend-modern/src/components/Settings/UpdatesSettingsPanel.tsx stays the
top-level settings shell, while
frontend-modern/src/components/Settings/UpdateInstallGuide.tsx,
frontend-modern/src/components/Settings/CopyCommandBlock.tsx, and
frontend-modern/src/components/Settings/updatesSettingsModel.ts plus
frontend-modern/src/utils/updatesPresentation.ts own the
deployment-specific install guide, copy-command block, and update-channel/install
model data plus customer-facing update status/action copy. The panel shell must
not rebuild copy-to-clipboard command cards, deployment instruction trees, or
update-surface wording inline.
The reporting operations surface now follows the same shell-state-model rule.
frontend-modern/src/components/Settings/ReportingPanel.tsx stays the
operations-panel shell, while
frontend-modern/src/components/Settings/useReportingPanelState.ts owns the
license/trial lifecycle and report generation flow,
frontend-modern/src/components/Settings/reportingPanelModel.ts plus
frontend-modern/src/utils/reportingResourceTypes.ts own the
request/range/filename model and reporting-type API mapping,
frontend-modern/src/components/Settings/ResourcePicker.tsx plus
frontend-modern/src/utils/reportableResourceTypes.ts own the reportable
resource selection, filter, sort, and empty-state contract, and
frontend-modern/src/utils/reportingPresentation.ts owns the user-facing
range/status copy. The compatibility re-export in
frontend-modern/src/components/Settings/reportingResourceTypes.ts stays part
of that same reporting boundary. The shell must not re-accumulate license
bootstrapping, inline report API requests, blob-download plumbing, or local
resource-type filter and reporting-token maps.
General settings segmented selectors for theme preference and temperature unit
must now also route through the shared FilterButtonGroup primitive instead of
maintaining local button-group styling forks inside
frontend-modern/src/components/Settings/GeneralSettingsPanel.tsx.
Reporting time-range/export selectors and General settings Proxmox VE polling
presets must now also route through the shared FilterButtonGroup prominent
variant instead of maintaining local blue segmented-control styling forks in
feature components.
That same shared FilterButtonGroup primitive must stay CSP-safe: touch-scroll
overflow behavior must come from canonical CSS classes rather than inline
style attributes so settings and reporting selectors do not reintroduce
browser console CSP violations under the release build policy.
Selectable settings cards for compact provider pickers and detail choice panels
must now route through the shared SelectionCardGroup primitive instead of
duplicating border-2 active-card styling in feature components.
Settings informational callouts with icon-plus-copy layouts must now route
through the shared CalloutCard primitive instead of maintaining feature-local
blue bordered wrappers.
Alert incident-event filter containers, labels, and chips must now route
through the shared presentation helpers in
frontend-modern/src/utils/alertIncidentPresentation.ts instead of allowing
frontend-modern/src/pages/Alerts.tsx and
frontend-modern/src/features/alerts/OverviewTab.tsx to fork their own filter
button styling.
Alert incident acknowledged badges, event cards, and note-editor controls must
also route through frontend-modern/src/utils/alertIncidentPresentation.ts
instead of letting the alerts page and overview timeline maintain duplicate
inline incident-detail styling.
Alert incident meta-row and detail-text presentation must also route through
frontend-modern/src/utils/alertIncidentPresentation.ts instead of letting
the alerts page and overview timeline maintain duplicated inline incident
typography rules.
Alert incident timeline event card structure must also route through
frontend-modern/src/components/Alerts/IncidentTimelineEventCard.tsx so the
alerts page and overview timeline share one canonical event-card renderer
instead of reimplementing the same summary/detail/output block twice.
The full expanded alert incident detail panel and event-filter controls must
also route through frontend-modern/src/components/Alerts/IncidentTimelinePanel.tsx
and frontend-modern/src/components/Alerts/IncidentEventFilters.tsx rather
than rebuilding loading/error copy, filter controls, note-editor wiring, or
event-card composition separately inside the alerts page and overview tab.
Resource incident panel card and summary-row presentation must also route
through frontend-modern/src/utils/alertIncidentPresentation.ts instead of
maintaining page-local incident panel styling inside
frontend-modern/src/pages/Alerts.tsx.
The settings shell now also has an explicit five-way ownership split.
frontend-modern/src/components/Settings/useDiscoverySettingsState.ts owns the
shared discovery draft and subnet-validation state,
frontend-modern/src/components/Settings/useSettingsInfrastructurePanelProps.ts
owns infrastructure workspace prop assembly and resource-derived infrastructure
read-model shaping for the shell,
frontend-modern/src/components/Settings/settingsNavigationModel.ts owns
settings tab identity, canonical route derivation, legacy alias normalization,
and Proxmox agent route metadata. settingsRouting.ts and
settingsTypes.ts remain thin compatibility re-export shims only, so external
consumers can bridge to the canonical owner without reintroducing a second
settings navigation model. settingsNavCatalog.ts owns settings navigation
metadata and item lookup, settingsNavVisibility.ts owns
feature/capability visibility and lock policy for settings navigation,
useSettingsNavigation.ts owns reactive URL sync and canonical tab-selection
state, SettingsDialogs.tsx owns shared settings modal composition,
including the route-owned billing focus contract where
/settings/system/billing/plan is the canonical settings-tab destination,
/settings/system/billing/usage is a same-tab child state, and legacy billing
base/hash links are compatibility inputs rather than primary runtime routes.
That same route-sync owner must also preserve Proxmox platform-selection truth
across canonical deep links such as
/settings/infrastructure/platforms/proxmox/pbs and
/settings/infrastructure/platforms/proxmox/pmg: even though those routes
collapse into the shared infrastructure-operations tab, the selected
platform state must still be derived from the path instead of silently
falling back to pve on reload or remount.
useSettingsShellState.ts owns shell-local sidebar/search/password-modal
state, and settingsTabSaveBehavior.ts owns settings tab save-behavior lookup,
frontend-modern/src/components/Settings/useSettingsSystemPanels.tsx owns
system panel prop assembly for general, network, updates, and recovery, and
frontend-modern/src/components/Settings/settingsPanelRegistryContext.tsx owns
registry context assembly for dispatchable settings tabs while
frontend-modern/src/components/Settings/settingsPanelRegistryLoaders.ts owns
the lazy settings panel loader table and route-to-panel import boundary, and
frontend-modern/src/components/Settings/useSettingsPanelRegistry.tsx owns the
final memoized registry composition only. frontend-modern/src/components/Settings/Settings.tsx
must stay a shell that wires those owners together instead of re-accumulating
infrastructure workspace props, registry context maps, system panel prop maps,
lazy loader definitions, or discovery draft state inline.
That same settings-routing contract now also owns the Support group for
Diagnostics & Health, Data & Reports, and System Logs: the navigation
model must normalize both /settings/operations/* and legacy /operations/*
compatibility links into /settings/support/*, and the catalog plus visibility
owners must treat those support surfaces as Settings-native pages rather than
as a second top-level utility destination.
The resource incident panel's collapsed activity summary is now part of that
same shared primitive boundary. Event-type count chips, visible-event copy,
and the summary-ordering helper in frontend-modern/src/features/alerts/types.ts
must stay shared across alert timeline surfaces instead of rebuilding
page-local event summaries or bespoke incident-card markup.
Feature-owned route surfaces under frontend-modern/src/features/ must also
keep their shell/runtime split explicit once a subsystem grows real transport
or polling lifecycle. The Patrol feature is the current reference shape:
frontend-modern/src/features/patrol/PatrolIntelligenceSurface.tsx stays the
feature shell, frontend-modern/src/features/patrol/usePatrolIntelligenceState.ts
owns the runtime state machine, frontend-modern/src/features/patrol/patrolInvestigationContextModel.ts
owns the pure investigation-context summary derivation,
frontend-modern/src/stores/aiIntelligenceSummaryModel.ts owns canonical AI
summary normalization at the shared store boundary, and the Patrol-owned
header/banner/summary/workspace section files under
frontend-modern/src/features/patrol/ own the heavy render surfaces. Shared
shell governance should reinforce that pattern instead of letting feature render
surfaces re-accumulate API and timer orchestration inline.
That same route-owned page-health rule now also applies to Patrol: a feature
surface may not present a green or all-clear primary summary when the owning
runtime contract says the page is blocked or unavailable, even if the last
successful snapshot was healthy.
That same rule also applies to compact Patrol summary fragments inside the
feature surface: count-only strips or metric cards must not emit No issues found or other reassuring copy when the owning overall-health summary is
degraded or not fully verified.
That same summary shell should also surface verification scope from the
owning run-history contract. Operators should be able to see, inside the same
summary surface, whether Patrol recently completed a full verification pass or
whether recent activity was limited to scoped/erroring patrol runs.
When the same governed run-history contract shows a recent full patrol plus
same-day scoped follow-up work, that summary shell should also carry a compact
activity-mix explanation rather than forcing operators to infer why Patrol
looked busy from a second competing status band.
That explanation belongs on the verification surface itself when operators are
reconciling Recently verified copy against same-day scoped Patrol bursts; the
supporting activity context may complement the readout, but it is not
sufficient as the only explanation path.
That same shell rule also owns Patrol recency labels. Shared Patrol header and
status-shell surfaces must keep Last full patrol tied only to the full-sweep
transport fact and use Last activity for scoped or verification work instead
of collapsing both timestamps back into a generic Last run label.
That same Patrol shell should make scoped trigger policy legible without
another navigation step. frontend-modern/src/features/patrol/PatrolIntelligenceHeader.tsx
should present alert-triggered and anomaly-triggered Patrol toggles as distinct
controls, and frontend-modern/src/components/patrol/PatrolStatusBar.tsx
should render compact activity breakdown and scoped-trigger-state copy from the
shared transport rather than leaving busy Patrol periods as unexplained noise.
That same Patrol-facing primitive vocabulary must stay product-first. Patrol
summary actions, runtime banners, circuit-breaker copy, and Patrol
configuration controls may point at the shared provider settings route or model
catalog, but they should describe those controls as Patrol/provider surfaces
rather than falling back to generic AI Settings, AI Model, or AI circuit breaker copy inside the Patrol shell itself.
That same product-first naming rule also applies to the shared system-ai
settings shell: frontend-modern/src/components/Settings/AISettings.tsx,
frontend-modern/src/components/Settings/settingsHeaderMeta.ts,
frontend-modern/src/components/Settings/settingsNavCatalog.ts,
frontend-modern/src/components/Settings/useAISettingsState.ts, and
frontend-modern/src/utils/aiSettingsPresentation.ts must present that surface
to operators as Assistant & Patrol plus provider/model configuration rather
than as a generic AI Services shell.
On the main Patrol page, though, that same governed activity context belongs
inside frontend-modern/src/features/patrol/PatrolIntelligenceSummary.tsx
alongside the verification readout rather than as a second full-width band
above the findings workspace. If PatrolStatusBar.tsx is reused elsewhere, it
must stay a compact factual support surface and must not reintroduce a parallel
page-level verdict strip once the summary shell already owns that explanation.
That same composition rule applies to frontend-modern/src/features/patrol/PatrolIntelligenceWorkspace.tsx:
once the summary shell carries the operator-facing verification and activity
story, the workspace should move directly into findings and run history instead
of repeating that same runtime context through a second pre-tab status strip.
Supporting context follows that same composition rule. Recent changes, learned
correlations, and policy coverage belong behind an explicitly secondary
supporting-context disclosure that only appears when Patrol has active
findings, degraded or incomplete verification, or a selected run that needs
explanation; healthy fully verified Patrol states must not advertise that
supporting evidence as a peer workflow. When that disclosure expands, the
workspace must explicitly label findings and run history as Patrol verification
evidence and frame the supporting cards as explanatory context rather than as a
fresh Patrol result. frontend-modern/src/features/patrol/patrolSupportingContextPresentation.ts
must own that disclosure copy and toggle wording so the Patrol workspace does
not regress into inline shell-local trust language.
Shared primitive consumers that split status-dot tone and status-text tone must now keep both values routed through the same exported presentation helper. Feature cards such as RAID status may not call shadow local aliases that drift from the canonical shared class/variant helpers.
Alert resource display labels used by the thresholds editor and alerts page
must now route through the shared helper in
frontend-modern/src/features/alerts/helpers.ts instead of rebuilding
resource display-name fallback chains inline. Governed resources must preserve
their canonical policy-aware label across grouped node headers, docker host
grouping, and saved override rows rather than collapsing back to raw names or
friendly-name truncation.
Shared search inputs must now keep their forwarded keyboard, blur, and clear handlers as explicit callable functions instead of relying on loose Solid event-handler unions. Shared search primitives still need to accept the real input/button event targets, but direct invocation inside the primitive must stay type-safe so consumers do not reintroduce union-call regressions while adding history, shortcut, or trailing-control behavior.
Shared shared-shell primitives that expose semantic title or value-level
onChange props must now explicitly omit the conflicting DOM attribute names
from their inherited HTML props. CalloutCard, FilterSegmentedControl, and
Subtabs may still forward ordinary div attributes, but their canonical API
must preserve JSX element titles and value-callback handlers instead of
widening back to raw DOM attribute unions.
Shared entitlement/migration warning banners that live under
frontend-modern/src/components/shared/ must also keep their counted fleet
surface on the Pulse Unified Agent term. Shared primitive copy may describe
legacy/API-connected resources separately, but it may not regress the primary
banner label or CTA text back to host-agent product language.
The self-hosted commercial paywall copy on those shared warning surfaces is
now also explicitly locked to monitored systems rather than agents. When a
shared banner or shared settings shell is explaining self-hosted plan caps,
the operator-facing commercial term must follow the monitored-system model even
if explicit legacy-v5 compatibility helpers still decode older alias fields at
import boundaries.
That same settings-shell framing must stay in customer language. Shared headers
and descriptions should talk about monitored-system limits, plan limits, and
subscription or license status instead of reviving legacy installed-agent
terms or vague internal nouns like allocation.
That banner boundary now also owns the canonical monitored-system naming
surface directly: the shared warning component path and exported symbol are
MonitoredSystemLimitWarningBanner, and future work may not reintroduce an
agent-era banner filename or component name as the primary primitive.
That shared monitored-system warning banner now also follows the shell/runtime/model
owner split. frontend-modern/src/components/shared/MonitoredSystemLimitWarningBanner.tsx
stays the render shell, frontend-modern/src/components/shared/useMonitoredSystemLimitWarningBannerState.ts
owns entitlement load, warning metric emission, migration/upgrade click tracking,
and upgrade-link runtime, and
frontend-modern/src/components/shared/monitoredSystemLimitWarningBannerModel.ts
owns monitored-system warning policy, count aggregation, and tone/text-class
policy while sourcing customer-facing monitored-system copy from the canonical
frontend-modern/src/utils/monitoredSystemPresentation.ts helper. Future
warning-banner work should extend those owners instead of pushing entitlement
state or route selection back into the render shell. When the warning points at
Pulse Pro billing, the shared primitive must stay a compact pointer rather
than re-expanding into a full policy explainer. The banner may signal the
current monitored-system posture and link into the owned billing surface, but
the longer over-plan or continuity explanation belongs in the plan-surface
capacity section owned by cloud-paid, not in permanent app-shell banner copy.
That same shared warning boundary now also owns the monitored-system capacity
posture vocabulary. Shared banners, plan summaries, and ledger headers must
describe the canonical admission-freeze model from
monitored_system_capacity: existing monitoring continues, new monitored
systems block at the plan boundary, and over-plan posture is an explicitly
frozen state. Shared primitives must not fall back to raw current / limit
slash math or 0 remaining wording that implies Pulse should retroactively
black out already-monitored systems.
orchestration, tracking, or naming math back into the shared shell or
reintroducing banner-local monitored-system copy strings.
Shared frontend label-formatting helpers now also have an explicit owner here.
frontend-modern/src/utils/textPresentation.ts is the canonical shared owner
for token humanization, identifier label formatting, title-casing, and
arrow-delimited label presentation used across AI, Patrol, Storage/Recovery,
and other feature surfaces. Feature contracts may depend on that helper, but
they should not re-home or fork those generic text-formatting rules into
feature-local utilities.
That same shared presentation boundary now also owns operator feedback and
shared table-label semantics. frontend-modern/src/components/Toast/Toast.tsx
stays the render shell for the global toast stack,
frontend-modern/src/utils/toast.ts owns the app-level trigger helper,
frontend-modern/src/utils/semanticTonePresentation.ts owns canonical toast
and diagnostics tone classes, frontend-modern/src/utils/emptyStatePresentation.ts
owns the shared empty-state tone styling consumed by EmptyState, and
frontend-modern/src/utils/typeColumnPresentation.ts owns the single
canonical type-column label used across dashboard and alert tables. Future
feedback, empty-state, or shared type-column work should extend those helpers
instead of reintroducing panel-local tone classes, app-local toast wiring, or
copy drift between tables.
First-session educational surfaces must also stay brief, flat, and model-led.
When Pulse needs to teach a user how a flow works, the primary on-screen
guidance should collapse to a few short descriptions of the real product
mental model instead of a logo wall, feature brochure, or verbose internal
mechanics dump. The runtime wizard itself now stays on the two-step
Welcome -> Security path, while the separate setup-completion preview owns
the brief three-step explanation: install the Unified Agent, get the first
Pulse resource, then layer on additional context.
The settings shell is now also a governed frontend primitive boundary.
frontend-modern/src/utils/settingsShellPresentation.ts now owns the
customer-facing settings-shell framing copy for navigation, search, loading,
and unsaved-change banners so SettingsPageShell.tsx stays a render shell
instead of re-accumulating product wording inline.
The alerts page shell now follows that same page-shell rule for feature tabs:
frontend-modern/src/pages/Alerts.tsx owns navigation and cross-surface
routing, while feature-owned tab surfaces such as
frontend-modern/src/features/alerts/tabs/DestinationsTab.tsx and
frontend-modern/src/features/alerts/tabs/HistoryTab.tsx plus
frontend-modern/src/features/alerts/tabs/ScheduleTab.tsx and
frontend-modern/src/features/alerts/tabs/ThresholdsTab.tsx own their
tab-local rendering and interaction logic. Future alert tab cleanup should
continue by extracting page-local tab blocks into feature modules rather than
expanding the top-level page file again, and history-table behavior or
thresholds-table adapter logic should stay feature-owned unless it graduates
into a shared primitive used by more than one alert surface.
Within that thresholds surface, frontend-modern/src/components/Alerts/ThresholdsTable.tsx
is now explicitly a shell consumer rather than the data or controller owner,
and the tab render owners live in
frontend-modern/src/components/Alerts/ThresholdsTableProxmoxTab.tsx,
frontend-modern/src/components/Alerts/ThresholdsTablePMGTab.tsx,
frontend-modern/src/components/Alerts/ThresholdsTableAgentsTab.tsx, and
frontend-modern/src/components/Alerts/ThresholdsTableDockerTab.tsx.
frontend-modern/src/features/alerts/thresholds/hooks/useThresholdsTableState.ts
owns the neutral thresholds sub-route contract:
/alerts/thresholds/infrastructure, /alerts/thresholds/systems,
/alerts/thresholds/mail-gateway, and /alerts/thresholds/containers.
Legacy /alerts/thresholds/proxmox and /alerts/thresholds/agents links
must redirect to the neutral infrastructure and systems routes so API-backed
platforms such as TrueNAS stay on canonical page language rather than
provider-specific aliases.
The infrastructure tab is itself now a shell that composes
frontend-modern/src/components/Alerts/ThresholdsTableProxmoxNodesSection.tsx,
frontend-modern/src/components/Alerts/ThresholdsTableProxmoxPBSSection.tsx,
frontend-modern/src/components/Alerts/ThresholdsTableProxmoxGuestsSection.tsx,
frontend-modern/src/components/Alerts/ThresholdsTableProxmoxGuestFilteringSection.tsx,
frontend-modern/src/components/Alerts/ThresholdsTableProxmoxBackupsSection.tsx,
frontend-modern/src/components/Alerts/ThresholdsTableProxmoxSnapshotsSection.tsx,
and frontend-modern/src/components/Alerts/ThresholdsTableProxmoxStorageSection.tsx
using the shared contract in
frontend-modern/src/features/alerts/thresholds/thresholdsTableSectionProps.ts.
Future infrastructure-thresholds presentation changes should extend those section
surfaces rather than restoring mixed JSX ownership to
frontend-modern/src/components/Alerts/ThresholdsTableProxmoxTab.tsx.
The Docker tab now follows that same composition pattern through
frontend-modern/src/components/Alerts/ThresholdsTableDockerIgnoredPrefixesSection.tsx,
frontend-modern/src/components/Alerts/ThresholdsTableDockerServiceGapSection.tsx,
frontend-modern/src/components/Alerts/ThresholdsTableDockerHostsSection.tsx,
and frontend-modern/src/components/Alerts/ThresholdsTableDockerContainersSection.tsx.
Future Docker thresholds presentation changes should extend those section
surfaces rather than restoring mixed JSX ownership to
frontend-modern/src/components/Alerts/ThresholdsTableDockerTab.tsx.
The systems tab now follows that same composition pattern through
frontend-modern/src/components/Alerts/ThresholdsTableAgentsResourcesSection.tsx
and frontend-modern/src/components/Alerts/ThresholdsTableAgentDisksSection.tsx.
Future systems-thresholds presentation changes should extend those section
surfaces rather than restoring mixed JSX ownership to
frontend-modern/src/components/Alerts/ThresholdsTableAgentsTab.tsx.
The thresholds tab adapter contract now lives in
frontend-modern/src/features/alerts/thresholds/thresholdsTabModel.ts, so
frontend-modern/src/features/alerts/tabs/ThresholdsTab.tsx stays a thin shell
instead of carrying a duplicate table adapter contract inline. That adapter
must bridge function-valued selectors and mutation props into
frontend-modern/src/components/Alerts/ThresholdsTable.tsx explicitly; spread-
based table prop adapters are not allowed here because they can collapse
function props on the live Solid surface and break thresholds runtime state.
Canonical threshold row shaping now routes through
frontend-modern/src/features/alerts/thresholds/thresholdsResourceModel.ts
plus the family-owned feature hooks
frontend-modern/src/features/alerts/thresholds/hooks/useThresholdsHostData.ts,
frontend-modern/src/features/alerts/thresholds/hooks/useThresholdsDockerData.ts,
frontend-modern/src/features/alerts/thresholds/hooks/useThresholdsGuestData.ts,
frontend-modern/src/features/alerts/thresholds/hooks/useThresholdsInfrastructureData.ts,
with frontend-modern/src/features/alerts/thresholds/hooks/useThresholdsData.ts
limited to composing them. Thresholds-table controller state lives in
frontend-modern/src/features/alerts/thresholds/hooks/useThresholdsTableState.ts,
so future cleanup should extend those feature hooks or tab owners instead of
rebuilding resource normalization, tab render surfaces, or thresholds-table
runtime state inside the shell component.
The alerts page now also applies the same shell-versus-feature rule to
configuration orchestration. frontend-modern/src/pages/Alerts.tsx is the page
shell, while frontend-modern/src/features/alerts/AlertsConfigurationSurface.tsx
is the feature shell. The canonical runtime owner is now
frontend-modern/src/features/alerts/useAlertsConfigurationState.ts for alert
config transport and org-switch reload orchestration,
frontend-modern/src/features/alerts/useAlertsConfigurationSnapshotState.ts
for the default-backed mutable configuration snapshot plus apply/capture/reset
ownership,
frontend-modern/src/features/alerts/alertsConfigurationModel.ts for config
normalization, factory defaults, docker-gap validation, and payload
serialization, frontend-modern/src/features/alerts/alertOverridesModel.ts
for override normalization and resource-backed projection. That shared
feature-model boundary must also canonicalize legacy shared-storage override
keys onto the current storage resource id before thresholds rows are derived,
so migrated Ceph/shared-datastore overrides survive the feature-shell path
instead of dropping out of the live editor, and
frontend-modern/src/features/alerts/useAlertOverridesState.ts
for reactive override state and thresholds-facing resource selectors, and
frontend-modern/src/features/alerts/alertDestinationsModel.ts for
destination config normalization and payload shaping, and
frontend-modern/src/features/alerts/useAlertDestinationsState.ts for
notification destination reload and persistence orchestration.
Within that alerts configuration runtime, canonical container-runtime projection
now belongs to alertOverridesModel.ts,
useAlertOverridesState.ts, and useAlertsConfigurationState.ts. The
thresholds Containers workspace must treat API-backed app-container
parents such as TrueNAS as first-class Container Runtimes, while Docker-only
controls in ThresholdsTableDockerTab.tsx remain gated to real
docker-host resources instead of leaking onto platform-managed runtimes.
frontend-modern/src/features/alerts/useAlertWebhookDestinationsState.ts now
owns webhook runtime, and
frontend-modern/src/components/Alerts/ResourceTable.tsx now follows the same
shell rule: the shell only picks desktop vs mobile render ownership and bulk-edit
composition, while
frontend-modern/src/components/Alerts/AlertResourceTableDesktop.tsx,
frontend-modern/src/components/Alerts/AlertResourceTableMobile.tsx, and
frontend-modern/src/components/Alerts/AlertResourceGroupHeader.tsx own the
render-heavy table/card/group-header surfaces. Shared runtime state remains in
frontend-modern/src/components/Alerts/useAlertResourceTableState.ts, shared row
rendering remains in
frontend-modern/src/components/Alerts/AlertResourceTableRow.tsx, and shared
metric normalization remains in
frontend-modern/src/components/Alerts/alertResourceTableModel.ts.
frontend-modern/src/features/alerts/useAlertDestinationsTabState.ts now owns
destination test actions plus retry orchestration while
frontend-modern/src/features/alerts/tabs/DestinationsTab.tsx stays the
render shell and should compose the dedicated email, Apprise, webhook, and
load/error section owners instead of carrying those panels inline. Future
cleanup should extend the transport hook, config model, override hook, or
destinations runtime hook based on the true owner, not move config control
flow back into the top-level page shell.
The alert email provider picker now also follows the shell/runtime split:
frontend-modern/src/components/Alerts/useEmailProviderSelectState.ts owns
provider-catalog loading and provider-default application, while
frontend-modern/src/components/Alerts/EmailProviderSelect.tsx stays the
render shell and should not re-accumulate NotificationsAPI.getEmailProviders
or a second local email-config contract inline.
The alert scheduling surface now follows the same shell-versus-section split:
frontend-modern/src/features/alerts/tabs/ScheduleTab.tsx should compose the
dedicated quiet-hours, cooldown, grouping, recovery, escalation, and summary
section owners while frontend-modern/src/features/alerts/useAlertScheduleState.ts
remains the canonical runtime owner.
The same rule now also covers cross-tab incident timelines: the shared runtime
owner is frontend-modern/src/features/alerts/useAlertIncidentTimelineState.ts,
while frontend-modern/src/features/alerts/OverviewTab.tsx and
frontend-modern/src/features/alerts/tabs/HistoryTab.tsx stay focused on
surface composition. Future incident timeline fetch, note-save, or expansion
control flow should extend that feature hook rather than forking back into
either tab surface.
Overview alert runtime now follows that same shell-versus-runtime split. The
shell stays in frontend-modern/src/features/alerts/OverviewTab.tsx, while
frontend-modern/src/features/alerts/useAlertOverviewState.ts owns derived
alert stats, filtered ordering, and single/bulk acknowledge runtime behavior.
Future overview control flow should extend that hook rather than restoring
action timers or acknowledge mutations to the tab shell.
Render-heavy overview ownership now lives in
frontend-modern/src/features/alerts/AlertOverviewStatsCards.tsx,
frontend-modern/src/features/alerts/AlertOverviewActiveAlertsSection.tsx,
and frontend-modern/src/features/alerts/AlertOverviewAlertCard.tsx, so
future card-list or timeline-card presentation work should extend those
surfaces rather than expanding frontend-modern/src/features/alerts/OverviewTab.tsx
back into a mixed shell.
Alert history runtime now follows that same pattern. The shell stays in
frontend-modern/src/features/alerts/tabs/HistoryTab.tsx, while
frontend-modern/src/features/alerts/useAlertHistoryState.ts owns history
fetch, persistent filters, history-clear behavior, and composition of the
derived history owners. Resource-incident panel runtime now lives in
frontend-modern/src/features/alerts/useAlertResourceIncidentsState.ts, while
frontend-modern/src/features/alerts/alertHistoryModel.ts owns grouped/trend
derivation and the bucket/range analytics contract. The render-heavy surfaces
now route through
frontend-modern/src/features/alerts/AlertHistoryFrequencyCard.tsx,
frontend-modern/src/features/alerts/AlertHistoryFiltersCard.tsx,
frontend-modern/src/features/alerts/AlertResourceIncidentsPanel.tsx,
frontend-modern/src/features/alerts/AlertHistoryTableSection.tsx,
frontend-modern/src/features/alerts/AlertHistoryTableGroupRow.tsx,
frontend-modern/src/features/alerts/AlertHistoryTableAlertRow.tsx, and
frontend-modern/src/features/alerts/AlertHistoryAdministrationCard.tsx.
Future alert-history control flow should extend the hook, pure history analytics
should extend the model, and section rendering should extend those owners
rather than rebuilding any of those concerns in the tab shell.
That same feature shell now owns the resource-resolution handoff into the
resource-incident panel. frontend-modern/src/features/alerts/tabs/HistoryTab.tsx
must pass the unified-resource resolver through to
frontend-modern/src/features/alerts/AlertResourceIncidentsPanel.tsx, and the
tab shell itself should only react to the current alertData() contract rather
than reviving deleted history-state aliases such as filteredAlerts(). The
panel may render compact route chips, but it must stay on shared route helpers
and feature-owned composition instead of growing provider-local routing logic
or another page-local resource lookup path.
Top-level settings surfaces must route through Settings.tsx,
SettingsPageShell.tsx, and
frontend-modern/src/components/shared/SettingsPanel.tsx instead of
reintroducing bespoke outer page headers or one-off top-level panel framing.
The shell metadata driving those surfaces is part of the same boundary as
well: frontend-modern/src/components/Settings/settingsHeaderMeta.ts and
representative top-level panels such as
frontend-modern/src/components/Settings/APIAccessPanel.tsx,
frontend-modern/src/components/Settings/AISettings.tsx,
frontend-modern/src/components/Settings/AIModelSelectionSection.tsx,
frontend-modern/src/components/Settings/AIRuntimeControlsSection.tsx,
frontend-modern/src/components/Settings/AIChatMaintenanceSection.tsx,
frontend-modern/src/components/Settings/AISettingsStatusAndActions.tsx,
frontend-modern/src/components/Settings/AIProviderConfigurationSection.tsx,
frontend-modern/src/components/Settings/AISettingsDialogs.tsx, and
frontend-modern/src/components/Settings/aiSettingsModel.ts now also define
the canonical AI settings runtime boundary. AISettings.tsx is the shell,
frontend-modern/src/components/Settings/useAISettingsState.ts owns the
runtime lifecycle and persistence flow, model/provider setup now routes
through AIModelSelectionSection.tsx, discovery, budget, timeout, and
permission controls route through AIRuntimeControlsSection.tsx, chat
maintenance routes through AIChatMaintenanceSection.tsx, and readiness plus
save/test actions route through AISettingsStatusAndActions.tsx. Future AI
settings work must extend those section owners instead of re-inlining large
runtime subsections into the shell.
That same AI settings boundary now also owns
frontend-modern/src/utils/aiSettingsPresentation.ts, so shared loading,
empty, OAuth, action/error, shell-description, and workload-discovery copy
for the settings shell stays on one governed helper instead of drifting back
into section-local strings.
frontend-modern/src/components/Settings/AuditLogPanel.tsx,
frontend-modern/src/components/Settings/AuditWebhookPanel.tsx,
frontend-modern/src/components/Settings/GeneralSettingsPanel.tsx,
frontend-modern/src/components/Settings/NetworkSettingsPanel.tsx,
frontend-modern/src/components/Settings/NetworkBoundarySettingsSection.tsx,
frontend-modern/src/components/Settings/networkSettingsModel.ts,
frontend-modern/src/components/Settings/SecurityAuthPanel.tsx,
frontend-modern/src/components/Settings/SecurityOverviewPanel.tsx,
frontend-modern/src/components/Settings/RecoverySettingsPanel.tsx,
frontend-modern/src/components/Settings/SSOProvidersPanel.tsx,
frontend-modern/src/components/Settings/useSSOProvidersState.ts, and
frontend-modern/src/components/Settings/ssoProvidersModel.ts now also define
the canonical SSO provider settings runtime boundary: SSOProvidersPanel.tsx
is the shell, useSSOProvidersState.ts owns the reactive/API lifecycle, and
ssoProvidersModel.ts owns provider-form normalization and payload building.
frontend-modern/src/components/Settings/UpdatesSettingsPanel.tsx must keep
page-shell titles, descriptions, and lead panel framing aligned instead of
letting navigation/header labels drift away from the actual settings surface.
The self-hosted Pulse Pro settings navigation item and route header metadata
for frontend-modern/src/components/Settings/settingsNavCatalog.ts and
frontend-modern/src/components/Settings/settingsHeaderMeta.ts are part of
that same shell boundary as
frontend-modern/src/components/Settings/ProLicensePanel.tsx and the shared
settings billing presentation owner in
frontend-modern/src/components/Settings/selfHostedBillingPresentation.ts;
the system-billing navigation label, header title/description, and billing
shell framing must all route through SELF_HOSTED_PRO_BILLING_PRESENTATION
instead of drifting independently. The owned split is now explicit: the
navigation label comes from navLabel, while the route header and billing
shell reuse shellTitle plus shellDescription, so the settings IA can stay
plan-owned (Plans) while the page itself still names the concrete job
(Plans & Activation) without reintroducing local label drift.
That same settings-shell framing boundary also covers adjacent top-level
settings references to the self-hosted commercial surface. When
InfrastructureWorkspace.tsx or other settings-shell surfaces point operators
toward Pulse Pro for billing, monitored-system limits, or license status, they
must reuse the shared referral copy from
SELF_HOSTED_PRO_BILLING_PRESENTATION rather than drafting local “go there
for billing” variants.
That same shared presentation owner now also carries the entitlement-first
commercial summary contract for self-hosted settings. The top-level navigation
entry stays product-IA owned through navLabel (Plans), while the page
header and shell title stay task-owned through shellTitle
(Plans & Activation), and the billing shell must foreground the active plan
name plus unlocked capabilities before secondary billing or recovery detail.
Paid upgrades should be able to confirm “Current plan: Pulse Pro” immediately
after activation without hunting through generic billing language or a second
page-local summary card model.
That same shell boundary also has to stay safe for hosted tenant bundles.
Settings-shell framing copy for self-hosted billing must route through
selfHostedBillingPresentation.ts, with settingsNavCatalog.ts,
settingsHeaderMeta.ts, and adjacent hosted settings shells consuming that
settings-owned adapter instead of importing generic commercial presentation
helpers in ways that can reintroduce top-level bundle-init cycles.
frontend-modern/src/components/Settings/NetworkSettingsPanel.tsx is now a
shell only for network-boundary controls.
frontend-modern/src/components/Settings/NetworkBoundarySettingsSection.tsx
owns the public URL, CORS, embedding, and webhook-boundary UI, while the
editable discovery configuration entry point is owned by the infrastructure
workspace instead of the System/Network route. Shared prop contracts for the
network-boundary surface must extend
frontend-modern/src/components/Settings/networkSettingsModel.ts instead of
re-expanding the shell or reintroducing page-local section types.
frontend-modern/src/utils/discoveryPresentation.ts now owns the
customer-facing discovery-section framing copy, scan-scope labels, subnet
guidance, and environment-lock messaging so
frontend-modern/src/components/Settings/DiscoverySettingsForm.tsx stays a
shared presentation shell instead of re-accumulating that wording inline.
That same settings-shell boundary now also owns the shared settings
presentation helpers that those panels consume. frontend-modern/src/utils/systemSettingsPresentation.ts
is the canonical owner for shared system-settings presets, summaries, and
customer-facing action copy, while
frontend-modern/src/utils/ssoProviderPresentation.ts owns the shared SSO
provider labels, empty states, and action/status messaging. Future settings
copy changes in those areas should extend these helpers instead of inlining
panel-local strings inside the shell or reactive state owners.
Shared infrastructure action-link framing now also owns recovery entry wording
for service resources. frontend-modern/src/components/Infrastructure/serviceDetailLinks.ts
must keep platform-service recovery links on canonical recovery-events
framing and route state, so upstream service surfaces do not drift back to
PBS-backup wording or inherit the page-default inventory workspace when they
are actually deep-linking into recovery activity.
That same shared primitive boundary also owns resource handoff chip framing for
cross-surface investigation UI. Alerts, Patrol, and similar feature shells may
choose which governed surfaces to show, but they must build those links through
the shared resolved-resource route helpers in
frontend-modern/src/routing/resourceLinks.ts instead of freezing raw route
strings, local link dedupe, or provider-specific link chips inside feature
panels. Shared chip styling belongs in the feature shell; canonical href and
label truth belongs in the shared route helper.
That same shared primitive boundary now also owns persisted column-identity
migration for governed surfaces. When a v6 surface canonicalizes saved column
IDs, frontend-modern/src/hooks/useColumnVisibility.ts must accept explicit
legacy-to-canonical aliases so existing local preferences migrate forward
without resetting user choices or forcing the runtime to keep deleted column
IDs alive indefinitely.
That same shared primitive boundary now also owns environment-lock
presentation. frontend-modern/src/components/shared/EnvironmentLockBadge.tsx
stays the reusable badge shell,
frontend-modern/src/utils/environmentLockPresentation.ts owns the canonical
badge label, title, and lock-button copy, and
frontend-modern/src/components/Settings/DockerRuntimeSettingsCard.tsx stays
the settings-shell consumer for environment-variable-locked container-update
controls. Future environment-lock UX should extend those owners instead of
reintroducing panel-local lock labels, badge styling, or title copy.
The release-ready shell proof now also includes a representative desktop
Playwright rehearsal in
tests/integration/tests/15-settings-shell-consistency.spec.ts so general,
organization, billing, relay, security, AI, updates, and recovery panels are
all exercised through the built app shell under a seeded multi-tenant runtime.
The security-facing settings panels within that shell now also follow an
explicit shared boundary with security-privacy so shell framing stays here
while auth posture, token controls, and privacy semantics remain governed as a
trust surface instead of generic UX copy.
That shared shell boundary now also covers version-matched docs-link framing:
customer-facing privacy disclosures in shared settings surfaces must route
through frontend-modern/src/utils/docsLinks.ts rather than panel-local
external URLs.
That same docs-link boundary also governs local legal docs surfaced from the
settings shell: shared settings surfaces such as
AIRuntimeControlsSection.tsx must route Terms-of-Service links through the
shipped TERMS.md asset instead of hardcoding GitHub main URLs that can
drift from the running build.
The same shell boundary now also owns shared relay route framing copy:
frontend-modern/src/utils/relayPresentation.ts is the canonical owner for
the top-level relay settings description and availability copy used by both
settingsHeaderMeta.ts and RelaySettingsPanel.tsx, so the route shell and
its first SettingsPanel cannot drift into separate rollout or pairing
descriptions.
Single-surface settings pages that only render one canonical SettingsPanel
must stay rooted directly at that panel instead of wrapping it in an extra
page-level space-y-* container. frontend-modern/src/components/Settings/UpdatesSettingsPanel.tsx
frontend-modern/src/components/Settings/RecoverySettingsPanel.tsx, and
frontend-modern/src/components/Settings/AuditLogPanel.tsx are the current
reference cases, and
frontend-modern/src/components/Settings/__tests__/settingsArchitecture.test.ts
locks that direct-root contract so single-surface pages do not quietly regain
redundant outer spacing chrome.
The same shared settings-shell boundary now also owns the API-backed
alternative path inside Infrastructure.
frontend-modern/src/components/Settings/InfrastructureWorkspace.tsx,
frontend-modern/src/components/Settings/settingsHeaderMeta.ts,
frontend-modern/src/components/Settings/settingsNavigationModel.ts,
frontend-modern/src/utils/dashboardEmptyStatePresentation.ts,
frontend-modern/src/utils/infrastructureEmptyStatePresentation.ts, and
adjacent setup guidance may still use Platform connections as the
operator-facing first-run label for API-backed onboarding, but that label must
resolve to the shared Infrastructure destination and its inline
ConnectionEditor add flow rather than reviving a standalone platform shell
or provider-local route.
That same settings-shell contract also owns the shared infrastructure summary
state. frontend-modern/src/components/Settings/useInfrastructureSettingsState.ts,
frontend-modern/src/components/Settings/useSettingsInfrastructurePanelProps.ts,
frontend-modern/src/components/Settings/InfrastructureWorkspace.tsx,
frontend-modern/src/components/Settings/useTrueNASSettingsPanelState.ts, and
frontend-modern/src/components/Settings/useVMwareSettingsPanelState.ts must
derive Proxmox/PBS/PMG/TrueNAS/VMware counts and availability from one shared
infrastructure settings state source instead of letting the top-level ledger
and inline credential flows fetch the same connection state separately. Phase
9 retired the standalone PlatformConnectionsWorkspace.tsx,
TrueNASSettingsPanel.tsx, and VMwareSettingsPanel.tsx shells; they remain
labels and proof history, not live presentation surfaces.
That same shared settings-shell boundary also owns provider parity inside the
inline add flow. Adding VMware may extend the same card, empty-state, dialog,
and summary-shell patterns used by TrueNAS, but it must not introduce a
VMware-only outer page shell, alternate settings route hierarchy, or another
summary vocabulary for connection health and contribution counts.
That same shared filter-presentation boundary also owns infrastructure
route-filter continuity. frontend-modern/src/features/infrastructure/
must keep a route-owned canonical source option such as truenas visible in
the shared LabeledFilterSelect even when current unified-resource results do
not contain that source, so platform handoffs from settings and other
surfaces do not flash back to All while the operator is still in a
provider-scoped investigation flow.
That same shared feature-presentation boundary also owns storage disk-detail
fallback messaging in frontend-modern/src/features/storageBackups/. Shared
detail presenters must describe the actual capability or identity gap that
prevents history from rendering, rather than reviving agent-install guidance
on API-backed platforms like TrueNAS when the canonical disk metrics target is
already the owning history path.
That same shared chart primitive boundary now also owns physical-disk live I/O
drawers. frontend-modern/src/components/Storage/DiskDetail.tsx must render
read, write, and busy charts through HistoryChart plus
useHistoryChartState, using the canonical physical-disk history resource id,
instead of reviving diskMetricsHistory, a page-local ring buffer, or another
storage-only live chart primitive for the same telemetry.
The shared shell boundary now also includes
frontend-modern/src/contexts/appRuntime.ts as the only neutral owner for
app-level websocket and dark-mode consumption. Shared shells and primitives
such as frontend-modern/src/components/Settings/Settings.tsx,
frontend-modern/src/components/shared/TagBadges.tsx, and
frontend-modern/src/components/shared/useInfrastructureSummaryTableState.ts
may consume that module, but they must not import @/App or recreate shell
providers. frontend-modern/src/App.tsx owns provider placement; primitives
own reusable consumption only.
That same shared settings-shell and banner boundary now also owns demo-mode
commercial suppression. frontend-modern/src/components/Settings/settingsNavCatalog.ts,
frontend-modern/src/components/Settings/settingsNavVisibility.ts,
frontend-modern/src/stores/sessionCapabilities.ts,
frontend-modern/src/stores/demoMode.ts,
frontend-modern/src/useAppRuntimeState.ts,
frontend-modern/src/components/shared/useTrialBannerState.ts, and
frontend-modern/src/components/shared/useMonitoredSystemLimitWarningBannerState.ts,
frontend-modern/src/components/shared/HistoryChartOverlay.tsx,
frontend-modern/src/features/patrol/PatrolIntelligenceBanners.tsx, and
frontend-modern/src/features/patrol/PatrolIntelligenceHeader.tsx
must consume one shared bootstrap truth from
/api/security/status.sessionCapabilities.demoMode and hide billing tabs,
trial nudges, monitored-system warning banners, dashboard upsells, Patrol
upgrade CTAs, history-lock paywalls, and other public-demo commercial
affordances when the browser is rendering a public demo runtime.
Shared primitives must not perform their own ad hoc /api/health polling,
response-header inference, hostname heuristics, or per-banner demo branching;
the runtime bootstrap, shared session-capability store, and shared banner
hooks stay on one canonical owner so suppression stays coherent across
customer-facing surfaces.
That same shared app-shell boundary now also owns assistant bootstrap silence
on non-AI routes. frontend-modern/src/useAppRuntimeState.ts,
frontend-modern/src/App.tsx,
frontend-modern/src/stores/aiChat.ts,
frontend-modern/src/components/AI/Chat/index.tsx, and
frontend-modern/src/hooks/useDashboardActions.ts must treat
/api/security/status.sessionCapabilities.assistantEnabled as the only
general-route assistant availability fact, while closed assistant chrome and
non-AI settings panels stay off /api/settings/ai and /api/ai/* until an
owned assistant or Patrol surface is actually open. frontend-modern/src/stores/aiChat.ts
must therefore stay presentation-only with respect to assistant bootstrap:
org-switch cleanup, keyboard focus, drawer state, and local context/session
persistence belong there, while backend settings/model reads stay on
frontend-modern/src/stores/aiRuntimeState.ts. The governed browser proof in
tests/integration/tests/11-first-session.spec.ts must continue to assert
that plain settings routes render without assistant bootstrap traffic or
console noise.
Shared table, disclosure, and form primitives must also stay explicitly typed
at the browser edge. Summary rows may memoize repeated pending-update reads,
shared buttons must preserve discriminated disclosure props, toggle and a11y
helpers must expose exact event signatures, shared rows must accept typed
data-* props, and reporting-panel helpers must remain ES2020-safe instead of
depending on feature-local casts or newer string helpers.
The settings navigation model now exposes a single infrastructure-systems
sidebar entry for the infrastructure settings area. The former
infrastructure-connections and infrastructure-install entries have been
removed from SettingsTab, settingsNavCatalog.ts, settingsPanelRegistry.ts,
and settingsNavigationModel.ts. All canonical redirects and tab-derivation
logic that previously mapped to those two entries now collapse to
infrastructure-systems. No future additions to the settings nav may restore
infrastructure-connections or infrastructure-install as independent tab
identifiers; panel routing within the infrastructure area must use
InfrastructurePanelStep in-page state instead of URL sub-routes.
frontend-modern/src/components/Settings/settingsNavigationModel.ts now uses
the normalised (not canonical) path when resolving Proxmox agent and path
checks so that deep links such as /settings/infrastructure/platforms/proxmox/pbs
resolve to the correct agent before the canonical-redirect fires, rather than
after it has already collapsed the path.