-
Notifications
You must be signed in to change notification settings - Fork 60
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Improve reliability of channel provider in case of reconnects (#1435)
* Abort loop earlier if possible * Reduce nesting * Swap the connection when ready * Explicit break to reduce nesting * Nullable enable * Move connection related stuff into the connection folder * Adjust the channel provider design slightly to achieve better testability without too much test induced damage * Test various races that can occur during shutdown and reconnection --------- Co-authored-by: Daniel Marbach <danielmarbach@users.noreply.github.com>
- Loading branch information
1 parent
d07d03e
commit 020c9e0
Showing
4 changed files
with
228 additions
and
19 deletions.
There are no files selected for viewing
168 changes: 168 additions & 0 deletions
168
src/NServiceBus.Transport.RabbitMQ.Tests/Connection/ChannelProviderTests.cs
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,168 @@ | ||
namespace NServiceBus.Transport.RabbitMQ.Tests.ConnectionString | ||
{ | ||
using System; | ||
using System.Collections.Generic; | ||
using System.Threading; | ||
using System.Threading.Tasks; | ||
using global::RabbitMQ.Client; | ||
using global::RabbitMQ.Client.Events; | ||
using NUnit.Framework; | ||
|
||
[TestFixture] | ||
public class ChannelProviderTests | ||
{ | ||
[Test] | ||
public async Task Should_recover_connection_and_dispose_old_one_when_connection_shutdown() | ||
{ | ||
var channelProvider = new TestableChannelProvider(); | ||
channelProvider.CreateConnection(); | ||
|
||
var publishConnection = channelProvider.PublishConnections.Dequeue(); | ||
publishConnection.RaiseConnectionShutdown(new ShutdownEventArgs(ShutdownInitiator.Library, 0, "Test")); | ||
|
||
channelProvider.DelayTaskCompletionSource.SetResult(); | ||
|
||
await channelProvider.FireAndForgetAction(CancellationToken.None); | ||
|
||
var recoveredConnection = channelProvider.PublishConnections.Dequeue(); | ||
|
||
Assert.That(publishConnection.WasDisposed, Is.True); | ||
Assert.That(recoveredConnection.WasDisposed, Is.False); | ||
} | ||
|
||
[Test] | ||
public void Should_dispose_connection_when_disposed() | ||
{ | ||
var channelProvider = new TestableChannelProvider(); | ||
channelProvider.CreateConnection(); | ||
|
||
var publishConnection = channelProvider.PublishConnections.Dequeue(); | ||
channelProvider.Dispose(); | ||
|
||
Assert.That(publishConnection.WasDisposed, Is.True); | ||
} | ||
|
||
[Test] | ||
public async Task Should_not_attempt_to_recover_during_dispose_when_retry_delay_still_pending() | ||
{ | ||
var channelProvider = new TestableChannelProvider(); | ||
channelProvider.CreateConnection(); | ||
|
||
var publishConnection = channelProvider.PublishConnections.Dequeue(); | ||
publishConnection.RaiseConnectionShutdown(new ShutdownEventArgs(ShutdownInitiator.Library, 0, "Test")); | ||
|
||
// Deliberately not completing the delay task with channelProvider.DelayTaskCompletionSource.SetResult(); before disposing | ||
// to simulate a pending delay task | ||
channelProvider.Dispose(); | ||
|
||
await channelProvider.FireAndForgetAction(CancellationToken.None); | ||
|
||
Assert.That(publishConnection.WasDisposed, Is.True); | ||
Assert.That(channelProvider.PublishConnections.TryDequeue(out _), Is.False); | ||
} | ||
|
||
[Test] | ||
public async Task Should_dispose_newly_established_connection() | ||
{ | ||
var channelProvider = new TestableChannelProvider(); | ||
channelProvider.CreateConnection(); | ||
|
||
var publishConnection = channelProvider.PublishConnections.Dequeue(); | ||
publishConnection.RaiseConnectionShutdown(new ShutdownEventArgs(ShutdownInitiator.Library, 0, "Test")); | ||
|
||
// This simulates the race of the reconnection loop being fired off with the delay task completed during | ||
// the disposal of the channel provider. To achieve that it is necessary to kick off the reconnection loop | ||
// and await its completion after the channel provider has been disposed. | ||
var fireAndForgetTask = channelProvider.FireAndForgetAction(CancellationToken.None); | ||
channelProvider.DelayTaskCompletionSource.SetResult(); | ||
channelProvider.Dispose(); | ||
|
||
await fireAndForgetTask; | ||
|
||
var recoveredConnection = channelProvider.PublishConnections.Dequeue(); | ||
|
||
Assert.That(publishConnection.WasDisposed, Is.True); | ||
Assert.That(recoveredConnection.WasDisposed, Is.True); | ||
} | ||
|
||
class TestableChannelProvider() : ChannelProvider(null!, TimeSpan.Zero, null!) | ||
{ | ||
public Queue<FakeConnection> PublishConnections { get; } = new(); | ||
|
||
public TaskCompletionSource DelayTaskCompletionSource { get; } = new(TaskCreationOptions.RunContinuationsAsynchronously); | ||
|
||
public Func<CancellationToken, Task> FireAndForgetAction { get; private set; } | ||
|
||
protected override IConnection CreatePublishConnection() | ||
{ | ||
var connection = new FakeConnection(); | ||
PublishConnections.Enqueue(connection); | ||
return connection; | ||
} | ||
|
||
protected override void FireAndForget(Func<CancellationToken, Task> action, CancellationToken cancellationToken = default) | ||
=> FireAndForgetAction = _ => action(cancellationToken); | ||
|
||
protected override async Task DelayReconnect(CancellationToken cancellationToken = default) | ||
{ | ||
await using var _ = cancellationToken.Register(() => DelayTaskCompletionSource.TrySetCanceled(cancellationToken)); | ||
await DelayTaskCompletionSource.Task; | ||
} | ||
} | ||
|
||
class FakeConnection : IConnection | ||
{ | ||
public int LocalPort { get; } | ||
public int RemotePort { get; } | ||
|
||
public void Dispose() => WasDisposed = true; | ||
|
||
public bool WasDisposed { get; private set; } | ||
|
||
public void UpdateSecret(string newSecret, string reason) => throw new NotImplementedException(); | ||
|
||
public void Abort() => throw new NotImplementedException(); | ||
|
||
public void Abort(ushort reasonCode, string reasonText) => throw new NotImplementedException(); | ||
|
||
public void Abort(TimeSpan timeout) => throw new NotImplementedException(); | ||
|
||
public void Abort(ushort reasonCode, string reasonText, TimeSpan timeout) => throw new NotImplementedException(); | ||
|
||
public void Close() => throw new NotImplementedException(); | ||
|
||
public void Close(ushort reasonCode, string reasonText) => throw new NotImplementedException(); | ||
|
||
public void Close(TimeSpan timeout) => throw new NotImplementedException(); | ||
|
||
public void Close(ushort reasonCode, string reasonText, TimeSpan timeout) => throw new NotImplementedException(); | ||
|
||
public IModel CreateModel() => throw new NotImplementedException(); | ||
|
||
public void HandleConnectionBlocked(string reason) => throw new NotImplementedException(); | ||
|
||
public void HandleConnectionUnblocked() => throw new NotImplementedException(); | ||
|
||
public ushort ChannelMax { get; } | ||
public IDictionary<string, object> ClientProperties { get; } | ||
public ShutdownEventArgs CloseReason { get; } | ||
public AmqpTcpEndpoint Endpoint { get; } | ||
public uint FrameMax { get; } | ||
public TimeSpan Heartbeat { get; } | ||
public bool IsOpen { get; } | ||
public AmqpTcpEndpoint[] KnownHosts { get; } | ||
public IProtocol Protocol { get; } | ||
public IDictionary<string, object> ServerProperties { get; } | ||
public IList<ShutdownReportEntry> ShutdownReport { get; } | ||
public string ClientProvidedName { get; } = $"FakeConnection{Interlocked.Increment(ref connectionCounter)}"; | ||
public event EventHandler<CallbackExceptionEventArgs> CallbackException = (_, _) => { }; | ||
public event EventHandler<ConnectionBlockedEventArgs> ConnectionBlocked = (_, _) => { }; | ||
public event EventHandler<ShutdownEventArgs> ConnectionShutdown = (_, _) => { }; | ||
public event EventHandler<EventArgs> ConnectionUnblocked = (_, _) => { }; | ||
|
||
public void RaiseConnectionShutdown(ShutdownEventArgs args) => ConnectionShutdown?.Invoke(this, args); | ||
|
||
static int connectionCounter; | ||
} | ||
} | ||
} |
File renamed without changes.
File renamed without changes.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters