23 KiB
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 ScadaLink 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:
DataTypeenum: 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.InstanceStateenum: Enabled, Disabled.DeploymentStatusenum: Pending, InProgress, Success, Failed.AlarmStateenum: Active, Normal.AlarmLevelenum: None, Low, LowLow, High, HighHigh. Severity level for an active alarm; alwaysNonefor binary trigger types, set byHiLotriggers.AlarmTriggerTypeenum: ValueMatch, RangeViolation, RateOfChange, HiLo.ConnectionHealthenum: 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 existingNotificationIdis the notification-specific name for this same concept.TrackedOperationKindenum: ExternalCall, DatabaseWrite. Discriminates the two cached-call kinds carried by a tracked operation (notifications are tracked separately via theNotificationTypeenum).TrackedOperationStatusenum: 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 ownStoreAndForwardMessageStatus, 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 (theNotificationStatusenum omitsFailed).
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.).- 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.AttributesasICollection<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,TemplateScript,TemplateComposition,TemplateFolder. - Instances:
Instance,InstanceAttributeOverride,InstanceConnectionBinding,InstanceAlarmOverride,Area. - Shared Scripts:
SharedScript. - Sites & Data Connections:
Site,DataConnection. - External Systems & Database Connections:
ExternalSystemDefinition,ExternalSystemMethod,DatabaseConnectionDefinition. - Notifications:
NotificationList(carries aTypefield —NotificationTypeenum — 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.
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, scripts, compositions, template folders, instances, overrides, alarm 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 theTypefield), recipients, SMTP configuration.INotificationOutboxRepository— TheNotificationstable: insert-if-not-exists ingest onNotificationId, due-row polling (Pendingrows andRetryingrows pastNextAttemptAt), status transitions, KPI aggregate queries, and the bulk delete of terminal rows used by the daily purge job.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 (synchronousCalland store-and-forwardCachedCall). 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'sRoute.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.
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.
- 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) andNotificationSubmitAck(central acknowledgement returned only after theNotificationsrow is persisted — ack-after-persist — which the site Store-and-Forward Engine waits on before clearing the buffered message).NotificationStatusQuery/NotificationStatusResponseback theNotify.Statusscript 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— theTrackedOperationKindenum (ExternalCall/DatabaseWrite) — target summary, status, retry count, last error, key timestamps, and source instance / script provenance) andCachedCallReconcileRequest/CachedCallReconcileResponse(cursor-based per-site pull of tracking rows changed since a cursor, used so missed telemetry self-heals). All three live in theIntegration/message folder and are subject to the additive-only evolution rules in REQ-COM-5a, since they cross the site→central version-skew boundary. - Parked Operation Command DTOs:
RetryParkedOperationandDiscardParkedOperation(central→site command/control messages keyed byTrackedOperationId, 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 theRemoteQuery/message folder alongside the other parked-message management messages.
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:
ScadaLink.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)
│ ├── Enums/ # InstanceState, DeploymentStatus, AlarmState,
│ │ # AlarmLevel, AlarmTriggerType, ConnectionHealth,
│ │ # DataType, StoreAndForwardCategory,
│ │ # StoreAndForwardMessageStatus,
│ │ # NotificationType, NotificationStatus,
│ │ # TrackedOperationKind, TrackedOperationStatus
│ ├── DataConnections/ # OPC UA endpoint config value objects + enums
│ ├── Flattening/ # FlattenedConfiguration, ConfigurationDiff,
│ │ # DeploymentPackage, ValidationResult
│ └── Scripts/ # AlarmContext, ScriptScope
├── Interfaces/ # Shared interfaces by concern
│ ├── Protocol/ # REQ-COM-2: Protocol abstraction (IDataConnection, etc.)
│ ├── Repositories/ # REQ-COM-4: Per-component repository interfaces
│ │ ├── ITemplateEngineRepository.cs
│ │ ├── IDeploymentManagerRepository.cs
│ │ ├── ISecurityRepository.cs
│ │ ├── IInboundApiRepository.cs
│ │ ├── IExternalSystemRepository.cs
│ │ ├── INotificationRepository.cs
│ │ ├── INotificationOutboxRepository.cs
│ │ ├── ISiteRepository.cs
│ │ └── ICentralUiRepository.cs
│ └── Services/ # REQ-COM-4a: Cross-cutting service interfaces
│ ├── IAuditService.cs
│ ├── IDatabaseGateway.cs
│ ├── IExternalSystemClient.cs
│ ├── IInstanceLocator.cs
│ └── INotificationDeliveryService.cs
├── Entities/ # REQ-COM-3: Domain entity POCOs, by domain area
│ ├── Templates/ # Template, TemplateAttribute, TemplateAlarm,
│ │ # TemplateScript, TemplateComposition, TemplateFolder
│ ├── Instances/ # Instance, InstanceAttributeOverride,
│ │ # InstanceConnectionBinding, InstanceAlarmOverride, 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
├── 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
│ ├── Instance/ # attribute get/set request/command messages
│ ├── Integration/ # external-integration call request/response,
│ │ # cached-call tracking telemetry + reconcile
│ ├── Notification/ # NotificationSubmit + ack,
│ │ # NotificationStatusQuery/Response
│ ├── InboundApi/ # Route.To() request messages
│ ├── RemoteQuery/ # event-log and parked-message query messages,
│ │ # parked-operation retry/discard commands
│ └── Management/ # HTTP/ClusterClient management commands + registry
├── Serialization/ # OpcUaEndpointConfigSerializer (typed↔legacy JSON)
└── Validators/ # OpcUaEndpointConfigValidator
Naming rules:
- Namespaces mirror the folder structure:
ScadaLink.Commons.Entities.Templates,ScadaLink.Commons.Interfaces.Repositories, etc. - Interface names use the
Iprefix:ITemplateEngineRepository,IAuditService,IDataConnection. - Entity classes are named after the domain concept (no suffixes like
EntityorModel):Template,Instance,Site. - Message contracts are named as commands, events, or responses:
DeployInstanceCommand,DeploymentStatusResponse,AttributeValueChanged. - Enums use singular names:
AlarmState, notAlarmStates.
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.