Files
Joseph Doherty f428804bef docs: close native-alarm spec gaps surfaced by docs audit
The native alarms feature merged with 7 component docs updated, but the
spec layer drifted: HighLevelReqs, Commons, and ManagementService had no
native-alarm coverage and the README table flagged it on only one row.
Add HighLevelReqs §3.4.2 (+ validation), document the Commons
types/entities/messages and the 7 ManagementService commands, sync the
README rows + link the TreeView sub-component, fix 2 broken plan links,
and drop the one-off native-alarms RESUME scratchpad.
2026-05-31 07:41:49 -04:00

37 KiB
Raw Permalink Blame History

Component: Commons

Purpose

The Commons component provides the shared foundation of data types, interfaces, enums, message contracts, data transfer objects, and persistence-ignorant domain entity classes used across all other ScadaBridge components. It ensures consistent type definitions for cross-component communication, data access, and eliminates duplication of common abstractions.

Location

Referenced by all component libraries and the Host.

Responsibilities

  • Define shared data types (enums, value objects, result types) used across multiple components.
  • Define persistence-ignorant domain entity classes (POCOs) representing all configuration database entities. These classes have no dependency on Entity Framework or any persistence framework — EF mapping is handled entirely by the Configuration Database component via Fluent API.
  • Define per-component repository interfaces that consuming components use for data access. Repository implementations are owned by the Configuration Database component.
  • Define protocol abstraction interfaces for the Data Connection Layer.
  • Define cross-component message contracts and DTOs for deployment, health, communication, instance lifecycle, and other inter-component data flows.
  • Contain no business logic — only data structures, interfaces, and enums.
  • Maintain minimal dependencies — only core .NET libraries; no Akka.NET, no ASP.NET, no Entity Framework.

Requirements

REQ-COM-1: Shared Data Type System

Commons must define shared primitive and utility types used across multiple components, including but not limited to:

  • DataType enum: Enumerates the data types supported by the system (e.g., Boolean, Int32, Float, Double, String, DateTime, Binary).
  • RetryPolicy: A record or immutable class describing retry behavior (max retries, fixed delay between retries).
  • Result<T>: A discriminated result type that represents either a success value or an error, enabling consistent error handling across component boundaries without exceptions.
  • InstanceState enum: Enabled, Disabled.
  • DeploymentStatus enum: Pending, InProgress, Success, Failed.
  • AlarmState enum: Active, Normal.
  • AlarmLevel enum: None, Low, LowLow, High, HighHigh. Severity level for an active alarm; always None for binary trigger types, set by HiLo triggers.
  • AlarmTriggerType enum: ValueMatch, RangeViolation, RateOfChange, HiLo.
  • AlarmKind enum: Computed, NativeOpcUa, NativeMxAccess. Discriminates how an alarm's state is produced — evaluated at the site by an AlarmActor from attribute triggers (Computed) vs. mirrored read-only from a native source (OPC UA Alarms & Conditions / MxAccess Gateway).
  • AlarmShelveState enum: Unshelved, OneShotShelved, TimedShelved, PermanentShelved. OPC UA Part 9 shelving sub-state of an alarm condition; mirrored read-only from the source. Computed alarms are always Unshelved.
  • AlarmTransitionKind enum: Snapshot, SnapshotComplete, Raise, Acknowledge, Clear, Retrigger, StateChange. Classifies a NativeAlarmTransition; Snapshot/SnapshotComplete carry the initial active-condition replay produced on every (re)subscribe so consumers can re-seed state.
  • AlarmConditionState: The unified, read-only alarm condition state. The OPC UA Part 9 sub-conditions are orthogonal (and MxAccess's ACTIVE / ACTIVE_ACKED / INACTIVE map cleanly onto them), so they are modeled as independent flags — Active, Acknowledged, Confirmed (bool?; null when not confirmable), Shelve (AlarmShelveState), Suppressed — plus a Severity on the unified 01000 scale. Computed alarms populate it from State + Priority; native alarms mirror it from the source. (Helper AlarmConditionStateFactory builds the computed-alarm projection.)
  • NativeAlarmTransition: The protocol-neutral alarm transition emitted by an IAlarmSubscribableConnection adapter — SourceReference (stable per-condition key), SourceObjectReference (owning source object, used for instance routing), AlarmTypeName, Kind (AlarmTransitionKind), Condition (AlarmConditionState), source taxonomy/operator/value metadata (Category, Description, Message, OperatorUser, OperatorComment, CurrentValue, LimitValue), and OriginalRaiseTime? / TransitionTime.
  • ConnectionHealth enum: Connected, Disconnected, Connecting, Error.
  • TrackedOperationId: A GUID identifying a tracked store-and-forward operation (ExternalSystem.CachedCall, Database.CachedWrite, Notify.Send). Generated caller-side at the site at call time, returned to the script as a tracking handle, and reused as the idempotency key for telemetry sent to central. The notification domain's existing NotificationId is the notification-specific name for this same concept.
  • TrackedOperationKind enum: ExternalCall, DatabaseWrite. Discriminates the two cached-call kinds carried by a tracked operation (notifications are tracked separately via the NotificationType enum).
  • TrackedOperationStatus enum: Pending, Retrying, Delivered, Parked, Failed, Discarded. The unified lifecycle state shared by all tracked store-and-forward operations. This is the operation's externally-observable lifecycle status in the site-local tracking table (the status record); it is related to but distinct from the S&F buffer's own StoreAndForwardMessageStatus, which tracks a buffered message's retry state within the buffer (the retry mechanism). Failed (permanent failure) has no notification analogue — notifications use only the other five states (the NotificationStatus enum omits Failed).
  • AuditChannel enum: ApiOutbound, DbOutbound, Notification, ApiInbound. Discriminates the script-trust-boundary channel that produced an AuditEvent. Owned by the Audit Log component.
  • AuditKind enum: ApiCall, ApiCallCached, DbWrite, DbWriteCached, NotifySend, NotifyDeliver, InboundRequest, InboundAuthFailure, CachedSubmit, CachedResolve. Channel-specific event kind — the valid Kind values for each AuditChannel are listed in the Audit Log component design (Component-AuditLog.md).
  • AuditStatus enum: Submitted, Forwarded, Attempted, Delivered, Failed, Parked, Discarded, Skipped. Lifecycle status of an audit event row; cached operations transit Submitted → Forwarded → Attempted → Delivered/Parked/Discarded. Skipped covers short-circuited (e.g. dry-run) actions that should still be audited.
  • AuditForwardState enum: Pending, Forwarded, Reconciled. Site-local SQLite flag governing the telemetry/reconciliation loop (set on a row but never sent to central).
  • AuditEvent: A record carrying every column of the central AuditLog row — EventId (GUID, idempotency key), OccurredAtUtc, IngestedAtUtc, Channel (AuditChannel), Kind (AuditKind), CorrelationId, ExecutionId, ParentExecutionId, SourceSiteId, SourceNode, SourceInstanceId, SourceScript, Actor, Target, Status (AuditStatus), HttpStatus, DurationMs, ErrorMessage, ErrorDetail, RequestSummary, ResponseSummary, PayloadTruncated, Extra — plus a site-only ForwardState (AuditForwardState) used by the site SQLite write-buffer's telemetry/reconciliation loop. IngestedAtUtc is unset at the site and stamped on central ingest. See Component-AuditLog.md for the persistence schema and ingest semantics.
  • SiteCall: A record carrying the central SiteCalls operational-mirror row — TrackedOperationId, SourceSiteId, SourceNode, Kind, Target, Status, RetryCount, key timestamps, and provenance — fed by site CachedCallTelemetry and the periodic reconciliation pull.

Types defined here must be immutable and thread-safe.

Timestamp convention: All timestamps throughout the system must use UTC (DateTime with DateTimeKind.Utc or DateTimeOffset with zero offset). This applies to all stored timestamps (SQLite, MS SQL, audit log entries), all message timestamps (attribute values, alarm state changes, health reports, event log entries, deployment records), and all wire-format timestamps (Akka remoting, Inbound API responses). Local time conversion, if needed, is a UI display concern only.

REQ-COM-2: Protocol Abstraction

Commons must define the protocol abstraction interfaces that the Data Connection Layer implements and other components consume:

  • IDataConnection: The common interface for reading, writing, and subscribing to device data regardless of the underlying protocol (OPC UA, custom legacy, etc.).
  • IAlarmSubscribableConnection: An optional capability interface an IDataConnection implementation may also implement when its source can mirror native alarms (OPC UA Alarms & Conditions, MxAccess Gateway). Exposes SubscribeAlarmsAsync(sourceReference, conditionFilter, callback, ct) (returns a subscription id; replays a snapshot of currently-active conditions on every (re)subscribe) and UnsubscribeAlarmsAsync(subscriptionId, ct). Transitions are delivered via the AlarmTransitionCallback delegate as protocol-neutral NativeAlarmTransition records. Mirrors the IBrowsableDataConnection capability-interface pattern; consumed by the Data Connection Layer's DataConnectionActor only.
  • Related types: Tag identifiers, read/write results, subscription callbacks, connection status enums, and quality codes.

These interfaces must not reference any specific protocol implementation.

REQ-COM-3: Domain Entity Classes (POCOs)

Commons must define persistence-ignorant POCO entity classes for all configuration database entities. These classes:

  • Are plain C# classes with properties — no EF attributes, no base classes from EF, no navigation property annotations.
  • May include navigation properties (e.g., Template.Attributes as ICollection<TemplateAttribute>) defined as plain collections. The Configuration Database component configures the relationships via Fluent API.
  • May include constructors that enforce invariants (e.g., required fields).
  • Must have no dependency on Entity Framework Core or any persistence library.

Entity classes are organized by domain area:

  • Template & Modeling: Template, TemplateAttribute, TemplateAlarm, TemplateNativeAlarmSource, TemplateScript, TemplateComposition, TemplateFolder. TemplateNativeAlarmSource is a read-only binding (Name, Description, ConnectionName, SourceReference, optional ConditionFilter, plus IsLocked/IsInherited/LockedInDerived lock flags) to a native alarm source; inheritance/lock semantics mirror TemplateAlarm.
  • Instances: Instance, InstanceAttributeOverride, InstanceConnectionBinding, InstanceAlarmOverride, InstanceNativeAlarmSourceOverride, Area. InstanceNativeAlarmSourceOverride retargets an inherited native alarm source per physical instance, keyed by SourceCanonicalName; its ConnectionNameOverride / SourceReferenceOverride / ConditionFilterOverride fields each apply only when non-null (null keeps the inherited value), mirroring InstanceAlarmOverride.
  • Shared Scripts: SharedScript.
  • Sites & Data Connections: Site, DataConnection.
  • External Systems & Database Connections: ExternalSystemDefinition, ExternalSystemMethod, DatabaseConnectionDefinition.
  • Notifications: NotificationList (carries a Type field — NotificationType enum — selecting the list's notification type and its type-specific targets), NotificationRecipient, SmtpConfiguration, Notification (the durable central-queue row — see below).
  • Inbound API: ApiKey, ApiMethod.
  • Security: LdapGroupMapping, SiteScopeRule.
  • Deployment: DeploymentRecord, SystemArtifactDeploymentRecord, DeployedConfigSnapshot.
  • Audit: AuditLogEntry (configuration-change audit, owned by Configuration Database), AuditEvent (centralized Audit Log row, see REQ-COM-1), SiteCall (SiteCalls operational-mirror row).

The Notification entity is the persistence-ignorant POCO for a row of the central Notifications table — the durable notification queue owned by the Notification Outbox. It is a plain class with properties for NotificationId (GUID, the idempotency key), Type (NotificationType enum discriminator), ListName, Subject, Body, TypeData (a JSON string — the type-agnostic extensibility hook), Status (NotificationStatus enum), RetryCount, LastError, ResolvedTargets, the provenance fields SourceSiteId / SourceInstanceId / SourceScript, and the UTC timestamps SiteEnqueuedAt, CreatedAt, LastAttemptAt, NextAttemptAt, DeliveredAt. As with every entity class it has no EF dependency; the Configuration Database component supplies the Fluent API mapping, value conversions, and indexes. The Type and Status enums (NotificationType: Email, Teams, …; NotificationStatus: Pending, Retrying, Delivered, Parked, Discarded) are defined under Types/Enums/ per REQ-COM-1.

REQ-COM-4: Per-Component Repository Interfaces

Commons must define repository interfaces that consuming components use for data access. Each interface is tailored to the data needs of its consuming component:

  • ITemplateEngineRepository — Templates, attributes, alarms, native alarm sources, scripts, compositions, template folders, instances, overrides, alarm overrides, native alarm source overrides, connection bindings, areas.
  • IDeploymentManagerRepository — Deployment records, deployed configuration snapshots, system-wide artifact deployment records.
  • ISecurityRepository — LDAP group mappings, site scoping rules.
  • IInboundApiRepository — API keys, API method definitions.
  • IExternalSystemRepository — External system definitions, method definitions, database connection definitions.
  • INotificationRepository — Notification lists (including the Type field), recipients, SMTP configuration.
  • INotificationOutboxRepository — The Notifications table: insert-if-not-exists ingest on NotificationId, due-row polling (Pending rows and Retrying rows past NextAttemptAt), status transitions, KPI aggregate queries, and the bulk delete of terminal rows used by the daily purge job.
  • ISiteCallAuditRepository — The SiteCalls table: insert-if-not-exists ingest on TrackedOperationId, upsert-on-newer-status from telemetry and reconciliation pulls, KPI aggregate queries, and the bulk delete of terminal rows used by the daily purge job.
  • IAuditLogRepository — The central AuditLog table (Audit Log #23): insert-if-not-exists ingest on EventId, keyset-paged query, monthly partition switch-out and boundary inspection, KPI snapshots, recursive execution-tree walks, and distinct-source-node enumeration.
  • ISiteRepository — Sites, data connections, and their site assignments.
  • ICentralUiRepository — Read-oriented queries spanning multiple domain areas for display purposes.

All repository interfaces must:

  • Accept and return the POCO entity classes defined in Commons.
  • Include a SaveChangesAsync() method (or equivalent) to support unit-of-work commit.
  • Have no dependency on Entity Framework Core — they are pure interfaces.

Implementations of these interfaces are owned by the Configuration Database component.

REQ-COM-4a: Cross-Cutting Service Interfaces

Commons must define service interfaces for cross-cutting concerns that multiple components consume:

  • IAuditService: Provides a single method for components to log audit entries: LogAsync(user, action, entityType, entityId, entityName, afterState). The implementation (owned by the Configuration Database component) serializes the state as JSON and adds the audit entry to the current unit-of-work transaction. Defined in Commons so any central component can call it without depending on the Configuration Database component directly.
  • IDatabaseGateway: Provides script-facing ADO.NET database access via named database connections. Implemented by the External System Gateway, consumed by the Site Runtime's script runtime context.
  • IExternalSystemClient: Provides script-facing invocation of external system HTTP APIs (synchronous Call and store-and-forward CachedCall). Implemented by the External System Gateway, consumed by the script runtime context.
  • IInstanceLocator: Resolves an instance unique name to its site identifier. Used by the Inbound API's Route.To() to determine the destination site.
  • INotificationDeliveryService: Sends notifications to a named notification list, routing transient failures to store-and-forward. Implemented by the Notification Service, consumed by the script runtime context.
  • IAuditWriter: Site-local hot-path interface for appending an AuditEvent to the site SQLite AuditLog: Task WriteAsync(AuditEvent evt, CancellationToken ct). Single durable INSERT, ForwardState = Pending. Consumed by the script-trust-boundary call paths (External System Gateway, Database layer, Store-and-Forward Engine). Implementation lives in the Audit Log component.
  • ICentralAuditWriter: Central direct-write interface for central-originated audit rows (Inbound API request completion, Notification Outbox dispatcher attempts/terminals): Task WriteAsync(AuditEvent evt, CancellationToken ct), with insert-if-not-exists semantics on EventId so retried handlers cannot produce duplicates. Implementation lives in the Audit Log component.
  • ISiteAuditQueue: Site-local queue handing off AuditEvent rows from the hot path to the gRPC telemetry forwarder. Implementation lives in the Audit Log component.
  • ICachedCallLifecycleObserver / ICachedCallTelemetryForwarder: Bridge between the Store-and-Forward Engine's cached-call lifecycle transitions and the central CachedCallTelemetry packet (combined audit + operational state). Implementations live in the Audit Log component.
  • INodeIdentityProvider: Resolves the writing node's SourceNode label (node-a / node-b / central-a / central-b) stamped on every audit row, notification, and site-call.
  • IOperationTrackingStore: Site-local SQLite-backed status record store for tracked store-and-forward operations (Tracking.Status(id)).
  • IPartitionMaintenance: Central monthly partition-switch / retention purge job hook used by the Audit Log partition maintenance service.

Bundle transport interfaces (IBundleExporter, IBundleImporter, IBundleSessionStore, IAuditCorrelationContext) live alongside the data types in Interfaces/Transport/ and are owned by the Transport component (#24); they are defined in Commons so other components (Configuration Database for audit correlation, Central UI for the import workflow) can depend on the abstraction without taking a Transport dependency.

These interfaces are defined in Commons so that consuming components depend only on the abstraction, not on the implementing component.

REQ-COM-5: Cross-Component Message Contracts

Commons must define the shared DTOs and message contracts used for inter-component communication, including:

  • Deployment DTOs: Configuration snapshots, deployment commands, deployment status, validation results.
  • Instance Lifecycle DTOs: Disable, enable, delete commands and responses.
  • Health DTOs: Health check results, site status reports, heartbeat messages. Includes script error rates and alarm evaluation error rates.
  • Communication DTOs: Site identity, connection state, routing metadata.
  • Attribute Stream DTOs: Attribute value change messages (instance name, attribute path, value, quality, timestamp) and alarm state change messages (instance name, alarm name, state, priority, timestamp) for the site-wide Akka stream. The alarm state change message (AlarmStateChanged) is additively enriched to carry both computed and native alarms on one shape: an AlarmKind discriminator, the unified AlarmConditionState, and native metadata (SourceReference, AlarmTypeName, Category, OperatorUser, OperatorComment, OriginalRaiseTime, CurrentValue, LimitValue) — defaulted/empty for computed alarms. Subject to the additive-only evolution rules in REQ-COM-5a, since it crosses the site→central gRPC stream.
  • Native Alarm DTOs (Messages/DataConnection/): the read-only native alarm mirror messages between the Site Runtime and the Data Connection Layer — SubscribeAlarmsRequest / SubscribeAlarmsResponse (subscribe a source binding; response carries success + optional error), UnsubscribeAlarmsRequest, NativeAlarmTransitionUpdate (ConnectionName, Transition — one routed NativeAlarmTransition, including snapshot replay), and NativeAlarmSourceUnavailable (ConnectionName, SourceReference, Timestamp — the feed dropped on connection loss).
  • Debug View DTOs: Subscribe/unsubscribe requests, one-shot snapshot request (DebugSnapshotRequest), initial snapshot, stream filter criteria.
  • Script Execution DTOs: Script call requests (with recursion depth), return values, error results.
  • System-Wide Artifact DTOs: Shared script packages, external system definitions, database connection definitions, notification list definitions.
  • Notification DTOs: NotificationSubmit (site→central submission: NotificationId, ListName, Subject, Body, provenance, SiteEnqueuedAt) and NotificationSubmitAck (central acknowledgement returned only after the Notifications row is persisted — ack-after-persist — which the site Store-and-Forward Engine waits on before clearing the buffered message). NotificationStatusQuery / NotificationStatusResponse back the Notify.Status script API, round-tripping a status record (status, retry count, last error, key timestamps) once a notification has been forwarded. Recipient resolution is not part of any contract — the site forwards only (listName, subject, body) and central resolves the list at delivery time. Subject to the additive-only evolution rules in REQ-COM-5a, since a submission can cross the site→central version-skew boundary.
  • Cached Call Tracking DTOs: CachedCallTelemetry (site→central lifecycle telemetry for a tracked cached call: TrackedOperationId, source site, Kind — the TrackedOperationKind enum (ExternalCall / DatabaseWrite) — target summary, status, retry count, last error, key timestamps, and source instance / script provenance) and CachedCallReconcileRequest / CachedCallReconcileResponse (cursor-based per-site pull of tracking rows changed since a cursor, used so missed telemetry self-heals). All three live in the Integration/ message folder and are subject to the additive-only evolution rules in REQ-COM-5a, since they cross the site→central version-skew boundary. CachedCallTelemetry is additively extended to also carry the AuditEvent content for the corresponding lifecycle transition (CachedEnqueued / CachedAttempt / CachedTerminal), so one packet drives both the SiteCalls operational upsert and the AuditLog insert-if-not-exists in a single central transaction — see Component-AuditLog.md, Cached Operations — Combined Telemetry.
  • Parked Operation Command DTOs: RetryParkedOperation and DiscardParkedOperation (central→site command/control messages keyed by TrackedOperationId, instructing the owning site to retry or discard a parked store-and-forward operation). These generalize the existing parked-message retry/discard commands to also cover parked cached calls; they live in the RemoteQuery/ message folder alongside the other parked-message management messages.
  • Audit Telemetry DTOs: AuditTelemetryEnvelope (site→central gRPC message wrapping a batch of AuditEvent rows for the IngestAuditEvents telemetry call) and the matching reconciliation pull messages (PullAuditEvents request/response carrying a sinceUtc cursor and a batch of AuditEvent rows). Live in the Integration/ message folder, subject to the additive-only evolution rules in REQ-COM-5a since they cross the site→central version-skew boundary. Cached-operation audit rows do not travel via AuditTelemetryEnvelope — they are folded into CachedCallTelemetry per the bullet above.

All message types must be record types or immutable classes suitable for use as Akka.NET messages (though Commons itself must not depend on Akka.NET).

REQ-COM-5a: Message Contract Versioning

Since the system supports cross-site artifact version skew (sites may temporarily run different versions), message contracts must follow additive-only evolution rules:

  • New fields may be added with default values. Existing fields must not be removed or have their types changed.
  • Serialization must tolerate unknown fields (forward compatibility) and missing optional fields (backward compatibility).
  • Breaking changes require a new message type and a coordinated deployment to all nodes.
  • The Akka.NET serialization binding configuration (in the Host component) must explicitly map message types to serializers to prevent accidental binary serialization.

REQ-COM-5b: Namespace & Folder Convention

All types in Commons are organized by category and domain area using a consistent namespace and folder hierarchy:

ZB.MOM.WW.ScadaBridge.Commons/
├── Types/                          # REQ-COM-1: Shared data types
│   ├── Result.cs
│   ├── RetryPolicy.cs
│   ├── ScriptArgs.cs               # script-call parameter normalization helper
│   ├── ScriptParameters.cs         # typed script-parameter access helper
│   ├── StaleTagMonitor.cs          # heartbeat staleness watchdog
│   ├── ValueFormatter.cs           # culture-invariant value-to-string helper
│   ├── DynamicJsonElement.cs       # dynamic JSON wrapper for scripts
│   ├── TrackedOperationId.cs       # tracked store-and-forward operation ID (GUID)
│   ├── AuditLogKpiSnapshot.cs      # central AuditLog KPI tile shape
│   ├── SiteAuditBacklogSnapshot.cs # per-site audit-forward backlog snapshot
│   ├── SiteCallOperational.cs      # SiteCalls operational-row projection
│   ├── TrackingStatusSnapshot.cs   # site-local Tracking.Status(id) projection
│   ├── Enums/                      # InstanceState, DeploymentStatus, AlarmState,
│   │                               #   AlarmLevel, AlarmTriggerType, AlarmKind,
│   │                               #   AlarmShelveState, AlarmTransitionKind,
│   │                               #   ConnectionHealth,
│   │                               #   DataType, StoreAndForwardCategory,
│   │                               #   StoreAndForwardMessageStatus,
│   │                               #   NotificationType, NotificationStatus,
│   │                               #   TrackedOperationKind, TrackedOperationStatus,
│   │                               #   AuditChannel, AuditKind, AuditStatus,
│   │                               #   AuditForwardState
│   ├── Alarms/                     # AlarmConditionState, AlarmConditionStateFactory,
│   │                               #   NativeAlarmTransition (unified read-only condition model)
│   ├── Audit/                      # AuditLogPaging, AuditLogQueryFilter,
│   │                               #   AuditQueryParamParsers, ExecutionTreeNode,
│   │                               #   SiteCallKpiSnapshot, SiteCallPaging,
│   │                               #   SiteCallQueryFilter, SiteCallSiteKpiSnapshot
│   ├── DataConnections/            # OPC UA endpoint config value objects + enums
│   ├── Flattening/                 # FlattenedConfiguration, ConfigurationDiff,
│   │                               #   DeploymentPackage, ValidationResult
│   ├── InboundApi/                 # ApiKeyHasher, ParameterDefinition
│   ├── Notifications/              # NotificationKpiSnapshot, NotificationOutboxFilter,
│   │                               #   SiteNotificationKpiSnapshot
│   ├── Transport/                  # Transport bundle value objects: BundleManifest,
│   │                               #   BundleSession, BundleSummary, EncryptionMetadata,
│   │                               #   ExportSelection, ImportPreview, ImportResolution,
│   │                               #   ImportResult, ManifestContentEntry
│   └── Scripts/                    # AlarmContext, ScriptScope
├── Interfaces/                     # Shared interfaces by concern
│   ├── IOperationTrackingStore.cs  # site-local tracked-operation status store
│   ├── IPartitionMaintenance.cs    # central partition-switch / retention purge hook
│   ├── Protocol/                   # REQ-COM-2: Protocol abstraction (IDataConnection,
│   │                               #   IBrowsableDataConnection, IAlarmSubscribableConnection,
│   │                               #   AlarmTransitionCallback)
│   ├── Repositories/               # REQ-COM-4: Per-component repository interfaces
│   │   ├── ITemplateEngineRepository.cs
│   │   ├── IDeploymentManagerRepository.cs
│   │   ├── ISecurityRepository.cs
│   │   ├── IInboundApiRepository.cs
│   │   ├── IExternalSystemRepository.cs
│   │   ├── INotificationRepository.cs
│   │   ├── INotificationOutboxRepository.cs
│   │   ├── ISiteCallAuditRepository.cs
│   │   ├── IAuditLogRepository.cs
│   │   ├── ISiteRepository.cs
│   │   └── ICentralUiRepository.cs
│   ├── Services/                   # REQ-COM-4a: Cross-cutting service interfaces
│   │   ├── IAuditService.cs
│   │   ├── IAuditWriter.cs
│   │   ├── ICentralAuditWriter.cs
│   │   ├── ISiteAuditQueue.cs
│   │   ├── ICachedCallLifecycleObserver.cs
│   │   ├── ICachedCallTelemetryForwarder.cs
│   │   ├── INodeIdentityProvider.cs
│   │   ├── IDatabaseGateway.cs
│   │   ├── IExternalSystemClient.cs
│   │   ├── IInstanceLocator.cs
│   │   └── INotificationDeliveryService.cs
│   └── Transport/                  # Bundle transport interfaces (Transport #24):
│       ├── IAuditCorrelationContext.cs
│       ├── IBundleExporter.cs
│       ├── IBundleImporter.cs
│       └── IBundleSessionStore.cs
├── Entities/                       # REQ-COM-3: Domain entity POCOs, by domain area
│   ├── Templates/                  # Template, TemplateAttribute, TemplateAlarm,
│   │                               #   TemplateNativeAlarmSource, TemplateScript,
│   │                               #   TemplateComposition, TemplateFolder
│   ├── Instances/                  # Instance, InstanceAttributeOverride,
│   │                               #   InstanceConnectionBinding, InstanceAlarmOverride,
│   │                               #   InstanceNativeAlarmSourceOverride, Area
│   ├── Sites/                      # Site, DataConnection
│   ├── ExternalSystems/            # ExternalSystemDefinition, ExternalSystemMethod,
│   │                               #   DatabaseConnectionDefinition
│   ├── Notifications/              # NotificationList, NotificationRecipient, SmtpConfiguration,
│   │                               #   Notification (central Notifications-table row)
│   ├── InboundApi/                 # ApiKey, ApiMethod
│   ├── Security/                   # LdapGroupMapping, SiteScopeRule
│   ├── Deployment/                 # DeploymentRecord, SystemArtifactDeploymentRecord,
│   │                               #   DeployedConfigSnapshot
│   ├── Scripts/                    # SharedScript
│   └── Audit/                      # AuditLogEntry (config-change audit),
│                                   #   AuditEvent (centralized AuditLog row),
│                                   #   SiteCall (SiteCalls operational mirror)
├── Messages/                       # REQ-COM-5: Cross-component message contracts, by concern
│   ├── Deployment/
│   ├── Lifecycle/
│   ├── Health/
│   ├── Communication/
│   ├── Streaming/
│   ├── DebugView/
│   ├── ScriptExecution/
│   ├── Artifacts/
│   ├── DataConnection/             # data-connection subscribe/write/health messages,
│   │                               #   native alarm subscribe/transition messages
│   ├── Instance/                   # attribute get/set request/command messages
│   ├── Integration/                # external-integration call request/response,
│   │                               #   cached-call tracking telemetry + reconcile,
│   │                               #   audit telemetry envelope + reconcile
│   ├── Notification/               # NotificationSubmit + ack,
│   │                               #   NotificationStatusQuery/Response
│   ├── InboundApi/                 # Route.To() request messages
│   ├── RemoteQuery/                # event-log and parked-message query messages,
│   │                               #   parked-operation retry/discard commands
│   ├── Audit/                      # Audit Log (#23) + Site Call Audit (#22) ingest:
│   │                               #   IngestAuditEventsCommand/Reply,
│   │                               #   IngestCachedTelemetryCommand/Reply,
│   │                               #   UpsertSiteCallCommand/Reply, SiteCallQueries,
│   │                               #   SiteCallRelayMessages
│   └── Management/                 # HTTP/ClusterClient management commands + registry,
│                                   #   including TransportCommands (Export/Preview/Import bundle)
├── Serialization/                  # OpcUaEndpointConfigSerializer (typed↔legacy JSON)
└── Validators/                     # OpcUaEndpointConfigValidator

Naming rules:

  • Namespaces mirror the folder structure: ZB.MOM.WW.ScadaBridge.Commons.Entities.Templates, ZB.MOM.WW.ScadaBridge.Commons.Interfaces.Repositories, etc.
  • Interface names use the I prefix: ITemplateEngineRepository, IAuditService, IDataConnection.
  • Entity classes are named after the domain concept (no suffixes like Entity or Model): Template, Instance, Site.
  • Message contracts are named as commands, events, or responses: DeployInstanceCommand, DeploymentStatusResponse, AttributeValueChanged.
  • Enums use singular names: AlarmState, not AlarmStates.

REQ-COM-6: No Business Logic; Pure Helpers Permitted

Commons must contain only:

  • Data structures (records, classes, structs)
  • Interfaces
  • Enums
  • Constants
  • Pure, stateless helpers — see the carve-out below.

It must not contain any business logic that orchestrates other components, service implementations that perform I/O (database, network, file system), actor definitions, or orchestration code.

Pure-helper carve-out. Commons may contain stateless, side-effect-free helper types whose behavior is confined to transforming, formatting, parsing, or validating the data types Commons already defines, provided they:

  • have no dependency on Akka.NET, ASP.NET Core, EF Core, or any I/O surface (consistent with REQ-COM-7);
  • hold no shared mutable state across calls (a self-contained instance helper such as StaleTagMonitor, which owns only its own timer, is acceptable);
  • do not call into other components or perform orchestration.

Examples currently in Commons that fall under this carve-out: Result<T>, ScriptParameters and ScriptArgs (script-parameter shaping), ValueFormatter (value-to-string formatting), DynamicJsonElement (dynamic JSON access), StaleTagMonitor (a self-contained heartbeat watchdog), OpcUaEndpointConfigSerializer (typed↔legacy JSON conversion of a Commons value object) and OpcUaEndpointConfigValidator (rule checks over a Commons value object). These are intentionally placed in Commons so every consuming component shares one implementation rather than duplicating the logic. Anything that would require an I/O dependency, mutable cross-call state, or knowledge of another component's behavior does not qualify and must live in the owning component.

REQ-COM-7: Minimal Dependencies

Commons must depend only on core .NET libraries (System.*, Microsoft.Extensions.Primitives if needed). It must not reference:

  • Akka.NET or any Akka.* packages
  • ASP.NET Core or any Microsoft.AspNetCore.* packages
  • Entity Framework Core or any Microsoft.EntityFrameworkCore.* packages
  • Any third-party libraries requiring paid licenses

This ensures Commons can be referenced by all components without introducing transitive dependency conflicts.


Dependencies

  • None — only core .NET SDK.

Interactions

  • All component libraries: Reference Commons for shared types, interfaces, entity classes, and contracts.
  • Configuration Database: Implements the repository interfaces defined in Commons. Maps the POCO entity classes to the database via EF Core Fluent API.
  • Host: References Commons transitively through the component libraries.