- Client.Shared-003: DefaultSessionAdapter.WriteValueAsync / CallMethodAsync guard against null/empty Results and throw ServiceResultException with the response's ServiceResult code instead of indexing into a missing list. - Client.Shared-004: DefaultSessionAdapter.CloseAsync / HistoryReadRawAsync / HistoryReadAggregateAsync use the Session.*Async overloads and honour the caller's CancellationToken. - Client.Shared-009: AcknowledgeAlarmAsync returns the underlying ServiceResultException.StatusCode on failure instead of always Good; IOpcUaClientService doc updated to describe the new contract. - Client.Shared-010: ConnectionSettings.CertificateStorePath defaults to empty; DefaultApplicationConfigurationFactory resolves the canonical PKI path lazily, so per-failover ConnectionSettings copies don't hit the filesystem. - Client.Shared-011: added the alarm-fallback regression test, extracted EndpointSelector as a pure static, and added EndpointSelectorTests covering security-mode match, Basic256Sha256 preference, fallback, diagnostics, hostname rewrite, and null/empty guards. Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
97 lines
3.0 KiB
C#
97 lines
3.0 KiB
C#
using Shouldly;
|
|
using Xunit;
|
|
using ZB.MOM.WW.OtOpcUa.Client.Shared.Models;
|
|
|
|
namespace ZB.MOM.WW.OtOpcUa.Client.Shared.Tests.Models;
|
|
|
|
public class ConnectionSettingsTests
|
|
{
|
|
[Fact]
|
|
public void Defaults_AreCorrect()
|
|
{
|
|
var settings = new ConnectionSettings();
|
|
|
|
settings.EndpointUrl.ShouldBe(string.Empty);
|
|
settings.FailoverUrls.ShouldBeNull();
|
|
settings.Username.ShouldBeNull();
|
|
settings.Password.ShouldBeNull();
|
|
settings.SecurityMode.ShouldBe(SecurityMode.None);
|
|
settings.SessionTimeoutSeconds.ShouldBe(60);
|
|
settings.AutoAcceptCertificates.ShouldBeTrue();
|
|
// CertificateStorePath defaults to empty so constructing settings does not
|
|
// touch disk; DefaultApplicationConfigurationFactory resolves the canonical
|
|
// PKI path lazily on first connect (Client.Shared-010).
|
|
settings.CertificateStorePath.ShouldBe(string.Empty);
|
|
}
|
|
|
|
[Fact]
|
|
public void Validate_ThrowsOnNullEndpointUrl()
|
|
{
|
|
var settings = new ConnectionSettings { EndpointUrl = null! };
|
|
Should.Throw<ArgumentException>(() => settings.Validate())
|
|
.ParamName.ShouldBe("EndpointUrl");
|
|
}
|
|
|
|
[Fact]
|
|
public void Validate_ThrowsOnEmptyEndpointUrl()
|
|
{
|
|
var settings = new ConnectionSettings { EndpointUrl = "" };
|
|
Should.Throw<ArgumentException>(() => settings.Validate())
|
|
.ParamName.ShouldBe("EndpointUrl");
|
|
}
|
|
|
|
[Fact]
|
|
public void Validate_ThrowsOnWhitespaceEndpointUrl()
|
|
{
|
|
var settings = new ConnectionSettings { EndpointUrl = " " };
|
|
Should.Throw<ArgumentException>(() => settings.Validate())
|
|
.ParamName.ShouldBe("EndpointUrl");
|
|
}
|
|
|
|
[Fact]
|
|
public void Validate_ThrowsOnZeroTimeout()
|
|
{
|
|
var settings = new ConnectionSettings
|
|
{
|
|
EndpointUrl = "opc.tcp://localhost:4840",
|
|
SessionTimeoutSeconds = 0
|
|
};
|
|
Should.Throw<ArgumentException>(() => settings.Validate())
|
|
.ParamName.ShouldBe("SessionTimeoutSeconds");
|
|
}
|
|
|
|
[Fact]
|
|
public void Validate_ThrowsOnNegativeTimeout()
|
|
{
|
|
var settings = new ConnectionSettings
|
|
{
|
|
EndpointUrl = "opc.tcp://localhost:4840",
|
|
SessionTimeoutSeconds = -1
|
|
};
|
|
Should.Throw<ArgumentException>(() => settings.Validate())
|
|
.ParamName.ShouldBe("SessionTimeoutSeconds");
|
|
}
|
|
|
|
[Fact]
|
|
public void Validate_ThrowsOnTimeoutAbove3600()
|
|
{
|
|
var settings = new ConnectionSettings
|
|
{
|
|
EndpointUrl = "opc.tcp://localhost:4840",
|
|
SessionTimeoutSeconds = 3601
|
|
};
|
|
Should.Throw<ArgumentException>(() => settings.Validate())
|
|
.ParamName.ShouldBe("SessionTimeoutSeconds");
|
|
}
|
|
|
|
[Fact]
|
|
public void Validate_SucceedsWithValidSettings()
|
|
{
|
|
var settings = new ConnectionSettings
|
|
{
|
|
EndpointUrl = "opc.tcp://localhost:4840",
|
|
SessionTimeoutSeconds = 120
|
|
};
|
|
Should.NotThrow(() => settings.Validate());
|
|
}
|
|
} |