feat(commons): Layer B serializer + validator handle new OPC UA settings

OpcUaEndpointConfigSerializer:
- ToFlatDict emits new scalar keys (DiscardOldest, SubscriptionPriority,
  SubscriptionDisplayName, TimestampsToReturn).
- ToFlatDict emits dotted sub-object keys (UserIdentity.TokenType / Username /
  Password / CertificatePath / CertificatePassword, Deadband.Type / Value)
  when those sub-objects are non-null.
- FromFlatDict reads the same keys back; missing keys preserve POCO defaults.
- Deadband.Value uses InvariantCulture for double parsing/formatting.

OpcUaEndpointConfigValidator:
- SubscriptionDisplayName required (non-empty).
- UserIdentity.UsernamePassword requires Username.
- UserIdentity.X509Certificate requires CertificatePath.
- Deadband.Value must be > 0 when Deadband is set.
- fieldPrefix propagates through sub-object error EntityNames.

Drives the 11 previously-failing tests green; 51/51 in the suite now pass.
This commit is contained in:
Joseph Doherty
2026-05-12 02:22:51 -04:00
parent 91450ec390
commit b60a8ef409
2 changed files with 68 additions and 0 deletions

View File

@@ -70,12 +70,29 @@ public static class OpcUaEndpointConfigSerializer
["KeepAliveCount"] = config.KeepAliveCount.ToString(),
["LifetimeCount"] = config.LifetimeCount.ToString(),
["MaxNotificationsPerPublish"] = config.MaxNotificationsPerPublish.ToString(),
["DiscardOldest"] = config.DiscardOldest.ToString(),
["SubscriptionPriority"] = config.SubscriptionPriority.ToString(),
["SubscriptionDisplayName"] = config.SubscriptionDisplayName,
["TimestampsToReturn"] = config.TimestampsToReturn.ToString(),
};
if (config.Heartbeat is { } hb)
{
dict["HeartbeatTagPath"] = hb.TagPath;
dict["HeartbeatMaxSilence"] = hb.MaxSilenceSeconds.ToString();
}
if (config.UserIdentity is { } ui)
{
dict["UserIdentity.TokenType"] = ui.TokenType.ToString();
dict["UserIdentity.Username"] = ui.Username;
dict["UserIdentity.Password"] = ui.Password;
dict["UserIdentity.CertificatePath"] = ui.CertificatePath;
dict["UserIdentity.CertificatePassword"] = ui.CertificatePassword;
}
if (config.Deadband is { } db)
{
dict["Deadband.Type"] = db.Type.ToString();
dict["Deadband.Value"] = db.Value.ToString(System.Globalization.CultureInfo.InvariantCulture);
}
return dict;
}
@@ -105,6 +122,16 @@ public static class OpcUaEndpointConfigSerializer
TryAssignInt(dict, "LifetimeCount", v => c.LifetimeCount = v);
TryAssignInt(dict, "MaxNotificationsPerPublish", v => c.MaxNotificationsPerPublish = v);
if (dict.TryGetValue("DiscardOldest", out var doStr) && bool.TryParse(doStr, out var doVal))
c.DiscardOldest = doVal;
if (dict.TryGetValue("SubscriptionPriority", out var spStr) && byte.TryParse(spStr, out var spVal))
c.SubscriptionPriority = spVal;
if (dict.TryGetValue("SubscriptionDisplayName", out var sdnStr) && !string.IsNullOrWhiteSpace(sdnStr))
c.SubscriptionDisplayName = sdnStr;
if (dict.TryGetValue("TimestampsToReturn", out var ttrStr)
&& Enum.TryParse<OpcUaTimestampsToReturn>(ttrStr, ignoreCase: true, out var ttr))
c.TimestampsToReturn = ttr;
if (dict.TryGetValue("HeartbeatTagPath", out var hbPath)
&& !string.IsNullOrWhiteSpace(hbPath))
{
@@ -113,6 +140,28 @@ public static class OpcUaEndpointConfigSerializer
c.Heartbeat = hb;
}
if (dict.TryGetValue("UserIdentity.TokenType", out var uiTt)
&& Enum.TryParse<OpcUaUserTokenType>(uiTt, ignoreCase: true, out var tokenType))
{
var ui = new OpcUaUserIdentityConfig { TokenType = tokenType };
if (dict.TryGetValue("UserIdentity.Username", out var u)) ui.Username = u;
if (dict.TryGetValue("UserIdentity.Password", out var p)) ui.Password = p;
if (dict.TryGetValue("UserIdentity.CertificatePath", out var cp)) ui.CertificatePath = cp;
if (dict.TryGetValue("UserIdentity.CertificatePassword", out var cpw)) ui.CertificatePassword = cpw;
c.UserIdentity = ui;
}
if (dict.TryGetValue("Deadband.Type", out var dbT)
&& Enum.TryParse<OpcUaDeadbandType>(dbT, ignoreCase: true, out var dbType))
{
var db = new OpcUaDeadbandConfig { Type = dbType };
if (dict.TryGetValue("Deadband.Value", out var dbV)
&& double.TryParse(dbV, System.Globalization.NumberStyles.Float,
System.Globalization.CultureInfo.InvariantCulture, out var dbVal))
db.Value = dbVal;
c.Deadband = db;
}
return c;
}

View File

@@ -40,6 +40,10 @@ public static class OpcUaEndpointConfigValidator
if (config.MaxNotificationsPerPublish < 1)
errors.Add(Err("MaxNotificationsPerPublish", "Must be ≥ 1."));
if (string.IsNullOrWhiteSpace(config.SubscriptionDisplayName))
errors.Add(Err("SubscriptionDisplayName",
"Subscription display name is required."));
if (config.Heartbeat is { } hb)
{
if (string.IsNullOrWhiteSpace(hb.TagPath))
@@ -49,6 +53,21 @@ public static class OpcUaEndpointConfigValidator
errors.Add(Err("Heartbeat.MaxSilenceSeconds", "Must be > 0."));
}
if (config.UserIdentity is { } ui)
{
if (ui.TokenType == OpcUaUserTokenType.UsernamePassword
&& string.IsNullOrWhiteSpace(ui.Username))
errors.Add(Err("UserIdentity.Username",
"Username is required when token type is UsernamePassword."));
if (ui.TokenType == OpcUaUserTokenType.X509Certificate
&& string.IsNullOrWhiteSpace(ui.CertificatePath))
errors.Add(Err("UserIdentity.CertificatePath",
"Certificate path is required when token type is X509Certificate."));
}
if (config.Deadband is { } db && db.Value <= 0)
errors.Add(Err("Deadband.Value", "Must be > 0."));
return errors.Count == 0
? ValidationResult.Success()
: ValidationResult.FromErrors(errors.ToArray());