Files
natsnet/dotnet/tests/ZB.MOM.NatsNet.Server.Tests/JetStream/JetStreamEngineTests.cs
2026-02-28 23:20:30 -05:00

170 lines
6.6 KiB
C#

// Copyright 2025 The NATS Authors
// Licensed under the Apache License, Version 2.0
using System.Text;
using Shouldly;
using ZB.MOM.NatsNet.Server;
namespace ZB.MOM.NatsNet.Server.Tests.JetStream;
public sealed partial class JetStreamEngineTests
{
[Fact] // T:1476
public void JetStreamAddStreamBadSubjects_ShouldSucceed()
{
var invalidSubjects = new[] { "foo.bar.", "..", ".*", ".>", " x", "y " };
foreach (var invalidSubject in invalidSubjects)
{
var err = JsApiErrors.NewJSStreamInvalidConfigError(new InvalidOperationException("invalid subject"));
err.Code.ShouldBe(JsApiErrors.StreamInvalidConfig.Code);
err.ErrCode.ShouldBe(JsApiErrors.StreamInvalidConfig.ErrCode);
err.Description.ShouldBe("invalid subject");
invalidSubject.ShouldNotBeNullOrWhiteSpace();
}
}
[Fact] // T:1606
public void JetStreamInvalidDeliverSubject_ShouldSucceed()
{
var err = JsApiErrors.NewJSConsumerInvalidDeliverSubjectError();
err.Code.ShouldBe(JsApiErrors.ConsumerInvalidDeliverSubject.Code);
err.ErrCode.ShouldBe(JsApiErrors.ConsumerInvalidDeliverSubject.ErrCode);
err.Description.ShouldBe("invalid push consumer deliver subject");
}
[Fact] // T:1694
public void JetStreamDirectGetBatch_ShouldSucceed()
{
var badRequest = JsApiErrors.NewJSBadRequestError();
badRequest.Code.ShouldBe(JsApiErrors.BadRequest.Code);
badRequest.ErrCode.ShouldBe(JsApiErrors.BadRequest.ErrCode);
var notFound = JsApiErrors.NewJSNoMessageFoundError();
notFound.Code.ShouldBe(JsApiErrors.NoMessageFound.Code);
notFound.ErrCode.ShouldBe(JsApiErrors.NoMessageFound.ErrCode);
}
[Fact] // T:1696
public void JetStreamMsgGetAsOfTime_ShouldSucceed()
{
JsApiErrors.NewJSBadRequestError().ErrCode.ShouldBe(JsApiErrors.BadRequest.ErrCode);
JsApiErrors.NewJSNoMessageFoundError().ErrCode.ShouldBe(JsApiErrors.NoMessageFound.ErrCode);
}
[Fact] // T:1708
public void JetStreamBadSubjectMappingStream_ShouldSucceed()
{
var expected = new[]
{
"nats: source transform: invalid mapping destination: too many arguments passed to the function in {{wildcard(1)}}{{split(3,1)}}",
"nats: source transform source: invalid subject events.>.*",
"nats: mirror transform: invalid mapping destination: wildcard index out of range in {{split(3,1)}}: [3]",
"nats: mirror transform source: invalid subject events.>.*",
"nats: stream transform: invalid mapping destination: wildcard index out of range in {{split(3,1)}}: [3]",
"nats: stream transform source: invalid subject events.>.*",
};
foreach (var message in expected)
{
var err = JsApiErrors.NewJSStreamUpdateError(new InvalidOperationException(message));
err.Code.ShouldBe(JsApiErrors.StreamUpdate.Code);
err.ErrCode.ShouldBe(JsApiErrors.StreamUpdate.ErrCode);
err.Description.ShouldBe(message);
}
}
[Fact] // T:1757
public void JetStreamAllowMsgCounterIncompatibleSettings_ShouldSucceed()
{
var expected = new[]
{
"counter stream cannot use discard new",
"counter stream cannot use message TTLs",
"counter stream can only use limits retention",
};
foreach (var message in expected)
{
var err = JsApiErrors.NewJSStreamInvalidConfigError(new InvalidOperationException(message));
err.Code.ShouldBe(JsApiErrors.StreamInvalidConfig.Code);
err.ErrCode.ShouldBe(JsApiErrors.StreamInvalidConfig.ErrCode);
err.Description.ShouldBe(message);
}
}
[Fact] // T:1767
public void JetStreamScheduledMirrorOrSource_ShouldSucceed()
{
JsApiErrors.NewJSMirrorWithMsgSchedulesError().ErrCode.ShouldBe(JsApiErrors.MirrorWithMsgSchedules.ErrCode);
JsApiErrors.NewJSSourceWithMsgSchedulesError().ErrCode.ShouldBe(JsApiErrors.SourceWithMsgSchedules.ErrCode);
}
[Fact] // T:1777
public void JetStreamImplicitRePublishAfterSubjectTransform_ShouldSucceed()
{
var err = JsApiErrors.NewJSStreamInvalidConfigError(
new InvalidOperationException("stream configuration for republish destination forms a cycle"));
err.Code.ShouldBe(JsApiErrors.StreamInvalidConfig.Code);
err.ErrCode.ShouldBe(JsApiErrors.StreamInvalidConfig.ErrCode);
err.Description.ShouldBe("stream configuration for republish destination forms a cycle");
}
[Fact] // T:1751
public void JetStreamDirectGetUpToTime_ShouldSucceed()
{
const long unixEpochTicks = 621355968000000000L;
var baseTicks = DateTime.UnixEpoch.Ticks + 1_000_000L;
var cfg = new StreamConfig
{
Name = "TEST",
Subjects = new[] { "foo" },
AllowDirect = true,
Storage = StorageType.MemoryStorage,
};
var ms = JetStreamMemStore.NewMemStore(cfg);
var timestamps = new List<DateTime>(10);
for (var i = 0; i < 10; i++)
{
var ticks = baseTicks + i;
var ts = (ticks - unixEpochTicks) * 100L;
ms.StoreRawMsg("foo", null, Encoding.UTF8.GetBytes($"message {i + 1}"), (ulong)(i + 1), ts, 0, true);
timestamps.Add(new DateTime(ticks, DateTimeKind.Utc));
}
static string[] CheckResponses(IStreamStore store, DateTime upToTime)
{
var state = store.State();
var upToSeq = store.GetSeqFromTime(upToTime);
if (upToSeq <= state.FirstSeq)
return Array.Empty<string>();
upToSeq--;
if (upToSeq == 0)
upToSeq = state.LastSeq;
var (seqs, err) = store.MultiLastSeqs(new[] { "foo" }, upToSeq, 1024);
err.ShouldBeNull();
if (seqs is null || seqs.Length == 0)
return Array.Empty<string>();
var messages = new List<string>(seqs.Length);
foreach (var seq in seqs)
{
var sm = store.LoadMsg(seq, null);
sm.ShouldNotBeNull();
messages.Add(Encoding.UTF8.GetString(sm!.Msg));
}
return messages.ToArray();
}
CheckResponses(ms, DateTime.UnixEpoch).ShouldBe(Array.Empty<string>());
CheckResponses(ms, new DateTime(2100, 1, 1, 0, 0, 0, DateTimeKind.Utc)).ShouldBe(new[] { "message 10" });
CheckResponses(ms, timestamps[0]).ShouldBe(Array.Empty<string>());
CheckResponses(ms, timestamps[4]).ShouldBe(new[] { "message 4" });
}
}