mirror of https://github.com/Squidex/squidex.git
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
160 lines
5.4 KiB
160 lines
5.4 KiB
// ==========================================================================
|
|
// Squidex Headless CMS
|
|
// ==========================================================================
|
|
// Copyright (c) Squidex UG (haftungsbeschraenkt)
|
|
// All rights reserved. Licensed under the MIT license.
|
|
// ==========================================================================
|
|
|
|
using System.Globalization;
|
|
using FakeItEasy;
|
|
using FluentAssertions;
|
|
using Xunit;
|
|
|
|
namespace Squidex.Infrastructure.EventSourcing
|
|
{
|
|
public class PollingSubscriptionTests
|
|
{
|
|
private readonly IEventStore eventStore = A.Fake<IEventStore>();
|
|
private readonly IEventSubscriber<StoredEvent> eventSubscriber = A.Fake<IEventSubscriber<StoredEvent>>();
|
|
private readonly string position = Guid.NewGuid().ToString();
|
|
private readonly string filter = "^my-stream";
|
|
|
|
[Fact]
|
|
public async Task Should_subscribe_on_start()
|
|
{
|
|
await SubscribeAsync(false);
|
|
|
|
A.CallTo(() => eventStore.QueryAllAsync(filter, position, A<int>._, A<CancellationToken>._))
|
|
.MustHaveHappenedOnceExactly();
|
|
}
|
|
|
|
[Fact]
|
|
public async Task Should_forward_exception_to_subscriber()
|
|
{
|
|
var ex = new InvalidOperationException();
|
|
|
|
A.CallTo(() => eventStore.QueryAllAsync(filter, position, A<int>._, A<CancellationToken>._))
|
|
.Throws(ex);
|
|
|
|
var sut = await SubscribeAsync(false);
|
|
|
|
A.CallTo(() => eventSubscriber.OnErrorAsync(sut, ex))
|
|
.MustHaveHappened();
|
|
}
|
|
|
|
[Fact]
|
|
public async Task Should_forward_operation_cancelled_exception_to_subscriber()
|
|
{
|
|
var ex = new OperationCanceledException();
|
|
|
|
A.CallTo(() => eventStore.QueryAllAsync(filter, position, A<int>._, A<CancellationToken>._))
|
|
.Throws(ex);
|
|
|
|
var sut = await SubscribeAsync(false);
|
|
|
|
A.CallTo(() => eventSubscriber.OnErrorAsync(sut, ex))
|
|
.MustHaveHappened();
|
|
}
|
|
|
|
[Fact]
|
|
public async Task Should_forward_aggregate_operation_cancelled_exception_to_subscriber()
|
|
{
|
|
var ex = new AggregateException(new OperationCanceledException());
|
|
|
|
A.CallTo(() => eventStore.QueryAllAsync(filter, position, A<int>._, A<CancellationToken>._))
|
|
.Throws(ex);
|
|
|
|
var sut = await SubscribeAsync(false);
|
|
|
|
A.CallTo(() => eventSubscriber.OnErrorAsync(sut, ex))
|
|
.MustHaveHappened();
|
|
}
|
|
|
|
[Fact]
|
|
public async Task Should_wake_up()
|
|
{
|
|
var sut = await SubscribeAsync(true);
|
|
|
|
A.CallTo(() => eventStore.QueryAllAsync(filter, A<string>._, A<int>._, A<CancellationToken>._))
|
|
.MustHaveHappened(2, Times.Exactly);
|
|
}
|
|
|
|
[Fact]
|
|
public async Task Should_forward_events_to_subscriber()
|
|
{
|
|
var events = Enumerable.Range(0, 50).Select(CreateEvent).ToArray();
|
|
|
|
var receivedEvents = new List<StoredEvent>();
|
|
|
|
A.CallTo(() => eventStore.QueryAllAsync(filter, position, A<int>._, A<CancellationToken>._))
|
|
.Returns(events.ToAsyncEnumerable());
|
|
|
|
A.CallTo(() => eventSubscriber.OnNextAsync(A<IEventSubscription>._, A<StoredEvent>._))
|
|
.Invokes(x => receivedEvents.Add(x.GetArgument<StoredEvent>(1)!));
|
|
|
|
await SubscribeAsync(true);
|
|
|
|
receivedEvents.Should().BeEquivalentTo(events);
|
|
}
|
|
|
|
[Fact]
|
|
public async Task Should_continue_on_last_position()
|
|
{
|
|
var events1 = Enumerable.Range(10, 10).Select(CreateEvent).ToArray();
|
|
var events2 = Enumerable.Range(20, 10).Select(CreateEvent).ToArray();
|
|
|
|
var lastPosition = events1[^1].EventPosition;
|
|
|
|
var receivedEvents = new List<StoredEvent>();
|
|
|
|
A.CallTo(() => eventStore.QueryAllAsync(filter, position, A<int>._, A<CancellationToken>._))
|
|
.Returns(events1.ToAsyncEnumerable());
|
|
|
|
A.CallTo(() => eventStore.QueryAllAsync(filter, lastPosition, A<int>._, A<CancellationToken>._))
|
|
.Returns(events2.ToAsyncEnumerable());
|
|
|
|
A.CallTo(() => eventSubscriber.OnNextAsync(A<IEventSubscription>._, A<StoredEvent>._))
|
|
.Invokes(x => receivedEvents.Add(x.GetArgument<StoredEvent>(1)!));
|
|
|
|
await SubscribeAsync(true);
|
|
|
|
receivedEvents.Should().BeEquivalentTo(events1.Union(events2));
|
|
}
|
|
|
|
private StoredEvent CreateEvent(int position)
|
|
{
|
|
return new StoredEvent(
|
|
"my-stream",
|
|
position.ToString(CultureInfo.InvariantCulture)!,
|
|
position,
|
|
new EventData(
|
|
"type",
|
|
new EnvelopeHeaders
|
|
{
|
|
[CommonHeaders.EventId] = Guid.NewGuid().ToString()
|
|
},
|
|
"payload"));
|
|
}
|
|
|
|
private async Task<IEventSubscription> SubscribeAsync(bool wakeup = true)
|
|
{
|
|
var sut = new PollingSubscription(eventStore, eventSubscriber, filter, position);
|
|
|
|
try
|
|
{
|
|
if (wakeup)
|
|
{
|
|
sut.WakeUp();
|
|
}
|
|
|
|
await Task.Delay(200);
|
|
}
|
|
finally
|
|
{
|
|
sut.Dispose();
|
|
}
|
|
|
|
return sut;
|
|
}
|
|
}
|
|
}
|
|
|