Group all 69 projects into category subfolders under src/ and tests/ so the Rider Solution Explorer mirrors the module structure. Folders: Core, Server, Drivers (with a nested Driver CLIs subfolder), Client, Tooling. - Move every project folder on disk with git mv (history preserved as renames). - Recompute relative paths in 57 .csproj files: cross-category ProjectReferences, the lib/ HintPath+None refs in Driver.Historian.Wonderware, and the external mxaccessgw refs in Driver.Galaxy and its test project. - Rebuild ZB.MOM.WW.OtOpcUa.slnx with nested solution folders. - Re-prefix project paths in functional scripts (e2e, compliance, smoke SQL, integration, install). Build green (0 errors); unit tests pass. Docs left for a separate pass. Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
95 lines
3.3 KiB
C#
95 lines
3.3 KiB
C#
using System;
|
|
using System.Linq;
|
|
using Shouldly;
|
|
using Xunit;
|
|
using ZB.MOM.WW.OtOpcUa.Driver.Historian.Wonderware.Backend;
|
|
|
|
namespace ZB.MOM.WW.OtOpcUa.Driver.Galaxy.Host.Tests
|
|
{
|
|
[Trait("Category", "Unit")]
|
|
public sealed class HistorianClusterEndpointPickerTests
|
|
{
|
|
private static HistorianConfiguration Config(params string[] nodes) => new()
|
|
{
|
|
ServerName = "ignored",
|
|
ServerNames = nodes.ToList(),
|
|
FailureCooldownSeconds = 60,
|
|
};
|
|
|
|
[Fact]
|
|
public void Single_node_config_falls_back_to_ServerName_when_ServerNames_empty()
|
|
{
|
|
var cfg = new HistorianConfiguration { ServerName = "only-node", ServerNames = new() };
|
|
var p = new HistorianClusterEndpointPicker(cfg);
|
|
p.NodeCount.ShouldBe(1);
|
|
p.GetHealthyNodes().ShouldBe(new[] { "only-node" });
|
|
}
|
|
|
|
[Fact]
|
|
public void Failed_node_enters_cooldown_and_is_skipped()
|
|
{
|
|
var now = new DateTime(2026, 4, 18, 10, 0, 0, DateTimeKind.Utc);
|
|
var p = new HistorianClusterEndpointPicker(Config("a", "b"), () => now);
|
|
|
|
p.MarkFailed("a", "boom");
|
|
p.GetHealthyNodes().ShouldBe(new[] { "b" });
|
|
}
|
|
|
|
[Fact]
|
|
public void Cooldown_expires_after_configured_window()
|
|
{
|
|
var clock = new DateTime(2026, 4, 18, 10, 0, 0, DateTimeKind.Utc);
|
|
var p = new HistorianClusterEndpointPicker(Config("a", "b"), () => clock);
|
|
p.MarkFailed("a", "boom");
|
|
p.GetHealthyNodes().ShouldBe(new[] { "b" });
|
|
|
|
clock = clock.AddSeconds(61);
|
|
p.GetHealthyNodes().ShouldBe(new[] { "a", "b" });
|
|
}
|
|
|
|
[Fact]
|
|
public void MarkHealthy_immediately_clears_cooldown()
|
|
{
|
|
var now = new DateTime(2026, 4, 18, 10, 0, 0, DateTimeKind.Utc);
|
|
var p = new HistorianClusterEndpointPicker(Config("a"), () => now);
|
|
p.MarkFailed("a", "boom");
|
|
p.GetHealthyNodes().ShouldBeEmpty();
|
|
p.MarkHealthy("a");
|
|
p.GetHealthyNodes().ShouldBe(new[] { "a" });
|
|
}
|
|
|
|
[Fact]
|
|
public void All_nodes_in_cooldown_returns_empty_healthy_list()
|
|
{
|
|
var now = new DateTime(2026, 4, 18, 10, 0, 0, DateTimeKind.Utc);
|
|
var p = new HistorianClusterEndpointPicker(Config("a", "b"), () => now);
|
|
p.MarkFailed("a", "x");
|
|
p.MarkFailed("b", "y");
|
|
p.GetHealthyNodes().ShouldBeEmpty();
|
|
p.NodeCount.ShouldBe(2);
|
|
}
|
|
|
|
[Fact]
|
|
public void Snapshot_reports_failure_count_and_last_error()
|
|
{
|
|
var now = new DateTime(2026, 4, 18, 10, 0, 0, DateTimeKind.Utc);
|
|
var p = new HistorianClusterEndpointPicker(Config("a"), () => now);
|
|
p.MarkFailed("a", "first");
|
|
p.MarkFailed("a", "second");
|
|
|
|
var snap = p.SnapshotNodeStates().Single();
|
|
snap.FailureCount.ShouldBe(2);
|
|
snap.LastError.ShouldBe("second");
|
|
snap.IsHealthy.ShouldBeFalse();
|
|
snap.CooldownUntil.ShouldNotBeNull();
|
|
}
|
|
|
|
[Fact]
|
|
public void Duplicate_hostnames_are_deduplicated_case_insensitively()
|
|
{
|
|
var p = new HistorianClusterEndpointPicker(Config("NodeA", "nodea", "NodeB"));
|
|
p.NodeCount.ShouldBe(2);
|
|
}
|
|
}
|
|
}
|