Headless CMS and Content Managment Hub
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

// ==========================================================================
// 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;
}
}
}