Files
natsnet/dotnet/tests/ZB.MOM.NatsNet.Server.Tests/ImplBacklog/RouteHandlerTests.Impltests.cs

235 lines
8.5 KiB
C#

using System.IO;
using System.Text;
using Shouldly;
using ZB.MOM.NatsNet.Server;
using ZB.MOM.NatsNet.Server.Internal;
namespace ZB.MOM.NatsNet.Server.Tests.ImplBacklog;
public sealed partial class RouteHandlerTests
{
[Fact] // T:2798
public void ClusterAdvertiseErrorOnStartup_ShouldSucceed()
{
var options = new ServerOptions();
options.Cluster.Advertise = "addr:::123";
var (server, err) = NatsServer.NewServer(options);
err.ShouldBeNull();
var startErr = server!.StartRouting();
startErr.ShouldNotBeNull();
startErr!.Message.ShouldContain("Cluster.Advertise");
}
[Fact] // T:2822
public void TLSRoutesCertificateImplicitAllowPass_ShouldSucceed()
{
var client = new ClientConnection(ClientKind.Router, nc: new MemoryStream());
client.MatchesPinnedCert(null).ShouldBeTrue();
}
[Fact] // T:2823
public void TLSRoutesCertificateImplicitAllowFail_ShouldSucceed()
{
var client = new ClientConnection(ClientKind.Router, nc: new MemoryStream());
var pinned = new PinnedCertSet([new string('a', 64)]);
client.MatchesPinnedCert(pinned).ShouldBeFalse();
}
[Fact] // T:2844
public void RouteParseOriginClusterMsgArgs_ShouldSucceed()
{
var c = new ClientConnection(ClientKind.Router)
{
Route = new Route { AccName = "MY_ACCOUNT"u8.ToArray() },
};
var err = c.ProcessRoutedOriginClusterMsgArgs("ORIGIN foo + bar queue1 queue2 12 345\r\n"u8.ToArray());
err.ShouldBeNull();
Encoding.ASCII.GetString(c.ParseCtx.Pa.Account!).ShouldBe("ORIGIN");
Encoding.ASCII.GetString(c.ParseCtx.Pa.Subject!).ShouldBe("foo");
Encoding.ASCII.GetString(c.ParseCtx.Pa.Reply!).ShouldBe("bar");
c.ParseCtx.Pa.Queues.ShouldNotBeNull();
c.ParseCtx.Pa.Queues!.Count.ShouldBe(3);
c.ParseCtx.Pa.Size.ShouldBe(345);
}
[Fact] // T:2850
public void RouteCompression_ShouldSucceed()
{
var opts = new ServerOptions();
opts.Cluster.Compression.Mode = CompressionMode.S2Fast;
var (server, err) = NatsServer.NewServer(opts);
err.ShouldBeNull();
server.ShouldNotBeNull();
var infoProto = server!.GenerateRouteInitialInfoJSON(string.Empty, CompressionMode.S2Fast, 0, GossipMode.Default);
infoProto.Length.ShouldBeGreaterThan(0);
Encoding.ASCII.GetString(infoProto).ShouldContain("\"compression\":\"s2_fast\"");
}
[Fact] // T:2819
public async Task RouteIPResolutionAndRouteToSelf_ShouldSucceed()
{
var (server, err) = NatsServer.NewServer(new ServerOptions());
err.ShouldBeNull();
server.ShouldNotBeNull();
var resolver = new FixedResolver(["127.0.0.1", "other.host.in.cluster"]);
var excluded = new HashSet<string>(StringComparer.Ordinal) { "127.0.0.1:1234" };
var (address, resolveErr) = await server!.GetRandomIP(resolver, "routehost:1234", excluded);
resolveErr.ShouldBeNull();
address.ShouldBe("other.host.in.cluster:1234");
}
[Fact]
public void NumRemotesInternal_WhenRoutesExist_ReturnsCount()
{
var (server, err) = NatsServer.NewServer(new ServerOptions());
err.ShouldBeNull();
server.ShouldNotBeNull();
var routesField = typeof(NatsServer).GetField("_routes", System.Reflection.BindingFlags.Instance | System.Reflection.BindingFlags.NonPublic);
routesField.ShouldNotBeNull();
routesField!.SetValue(
server,
new Dictionary<string, List<ClientConnection>>
{
["one"] = [],
["two"] = [],
});
var method = typeof(NatsServer).GetMethod("NumRemotesInternal", System.Reflection.BindingFlags.Instance | System.Reflection.BindingFlags.NonPublic);
method.ShouldNotBeNull();
var count = (int)method!.Invoke(server, null)!;
count.ShouldBe(2);
}
[Fact] // T:2854
public void RouteCompressionAuto_ShouldSucceed()
{
var errors = new List<Exception>();
var warnings = new List<Exception>();
var options = new ServerOptions();
var parseError = ServerOptions.ParseCluster(
new Dictionary<string, object?>
{
["name"] = "local",
["compression"] = new Dictionary<string, object?>
{
["mode"] = CompressionModes.S2Auto,
["rtt_thresholds"] = new List<object?> { "100ms", "200ms", "300ms" },
},
},
options,
errors,
warnings);
parseError.ShouldBeNull();
errors.ShouldBeEmpty();
options.Cluster.Compression.Mode.ShouldBe(CompressionModes.S2Auto);
options.Cluster.Compression.RttThresholds.Count.ShouldBe(3);
options.Cluster.Compression.RttThresholds[0].ShouldBe(TimeSpan.FromMilliseconds(100));
options.Cluster.Compression.RttThresholds[1].ShouldBe(TimeSpan.FromMilliseconds(200));
options.Cluster.Compression.RttThresholds[2].ShouldBe(TimeSpan.FromMilliseconds(300));
options = new ServerOptions();
errors.Clear();
warnings.Clear();
parseError = ServerOptions.ParseCluster(
new Dictionary<string, object?>
{
["compression"] = new Dictionary<string, object?>
{
["mode"] = CompressionModes.S2Auto,
["rtt_thresholds"] = new List<object?> { "0ms", "100ms", "0ms", "300ms" },
},
},
options,
errors,
warnings);
parseError.ShouldBeNull();
errors.ShouldBeEmpty();
options.Cluster.Compression.RttThresholds.Count.ShouldBe(4);
options.Cluster.Compression.RttThresholds[0].ShouldBe(TimeSpan.Zero);
options.Cluster.Compression.RttThresholds[1].ShouldBe(TimeSpan.FromMilliseconds(100));
options.Cluster.Compression.RttThresholds[2].ShouldBe(TimeSpan.Zero);
options.Cluster.Compression.RttThresholds[3].ShouldBe(TimeSpan.FromMilliseconds(300));
options = new ServerOptions();
errors.Clear();
warnings.Clear();
parseError = ServerOptions.ParseCluster(
new Dictionary<string, object?>
{
["compression"] = false,
},
options,
errors,
warnings);
parseError.ShouldBeNull();
errors.ShouldBeEmpty();
options.Cluster.Compression.Mode.ShouldBe(CompressionModes.Off);
}
[Fact] // T:2859
public void RouteSlowConsumerRecover_ShouldSucceed()
{
var (server, err) = NatsServer.NewServer(new ServerOptions());
err.ShouldBeNull();
using var outStream = new MemoryStream();
var route = new ClientConnection(ClientKind.Router, server, outStream)
{
OutWtp = WriteTimeoutPolicy.Retry,
OutMp = 1024 * 1024,
};
// Detect slow consumer state from write-timeout path.
route.HandleWriteTimeout(written: 1, attempted: 1024, numChunks: 2).ShouldBeFalse();
route.Flags.IsSet(ClientFlags.IsSlowConsumer).ShouldBeTrue();
// A successful flush should clear slow-consumer marker (recovered).
route.QueueOutbound("MSG test 1 5\r\nhello\r\n"u8.ToArray());
route.FlushOutbound().ShouldBeTrue();
route.Flags.IsSet(ClientFlags.IsSlowConsumer).ShouldBeFalse();
}
private sealed class FixedResolver(string[] hosts) : INetResolver
{
public Task<string[]> LookupHostAsync(string host, CancellationToken ct = default)
=> Task.FromResult(hosts);
}
[Fact] // T:2825
public void ClusterQueueGroupWeightTrackingLeak_ShouldSucceed()
{
var account = Account.NewAccount("$G");
var queueSub = new Subscription
{
Subject = "foo"u8.ToArray(),
Queue = "bar"u8.ToArray(),
Qw = 1,
};
account.UpdateLeafNodes(queueSub, 1);
var lqwsField = typeof(Account).GetField("_lqws", System.Reflection.BindingFlags.Instance | System.Reflection.BindingFlags.NonPublic);
lqwsField.ShouldNotBeNull();
var lqws = (Dictionary<string, int>?)lqwsField!.GetValue(account);
lqws.ShouldNotBeNull();
lqws!.TryGetValue("foo bar", out var initialWeight).ShouldBeTrue();
initialWeight.ShouldBe(1);
account.UpdateLeafNodes(queueSub, -1);
lqws = (Dictionary<string, int>?)lqwsField.GetValue(account);
(lqws?.ContainsKey("foo bar") ?? false).ShouldBeFalse();
}
}