Skip to content

Commit 437e1ff

Browse files
authored
Fix: Respect IReconnectRetryPolicy in disconnect loops (#2853)
Fixes the case where we loop from `Connecting` -> `BeginConnectAsync` -> `OnDisconnected` and the next heartbeat triggers a reconnect again due to the backoff thresholds resetting. This ultimately causes the backoff to not be respected. Here, we're doing 2 things: - Upping the max exponential backoff window to 60 seconds which better handlers clients at scale and mass DDoS cases. - Only resets the reconnect counter on the first disconnect, not doing so again until we've successfully reconnected and heartbeated again, so backoffs are properly respected.
1 parent 3bd3e66 commit 437e1ff

File tree

3 files changed

+14
-3
lines changed

3 files changed

+14
-3
lines changed

docs/ReleaseNotes.md

+2-1
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,8 @@ Current package versions:
77
| [![StackExchange.Redis](https://img.shields.io/nuget/v/StackExchange.Redis.svg)](https://www.nuget.org/packages/StackExchange.Redis/) | [![StackExchange.Redis](https://img.shields.io/nuget/vpre/StackExchange.Redis.svg)](https://www.nuget.org/packages/StackExchange.Redis/) | [![StackExchange.Redis MyGet](https://img.shields.io/myget/stackoverflow/vpre/StackExchange.Redis.svg)](https://www.myget.org/feed/stackoverflow/package/nuget/StackExchange.Redis) |
88

99
## Unreleased
10-
No pending unreleased changes.
10+
- Fix: Respect `IReconnectRetryPolicy` timing in the case that a node that was present disconnects indefinitely ([#2853 by NickCraver](https://github.com/StackExchange/StackExchange.Redis/pull/2853))
11+
- Changes max default retry policy backoff to 60 seconds ([#2853 by NickCraver](https://github.com/StackExchange/StackExchange.Redis/pull/2853))
1112

1213
## 2.8.24
1314

src/StackExchange.Redis/ExponentialRetry.cs

+1-1
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,7 @@ namespace StackExchange.Redis
88
public class ExponentialRetry : IReconnectRetryPolicy
99
{
1010
private readonly int deltaBackOffMilliseconds;
11-
private readonly int maxDeltaBackOffMilliseconds = (int)TimeSpan.FromSeconds(10).TotalMilliseconds;
11+
private readonly int maxDeltaBackOffMilliseconds = (int)TimeSpan.FromSeconds(60).TotalMilliseconds;
1212
[ThreadStatic]
1313
private static Random? r;
1414

src/StackExchange.Redis/PhysicalBridge.cs

+11-1
Original file line numberDiff line numberDiff line change
@@ -47,6 +47,7 @@ internal sealed class PhysicalBridge : IDisposable
4747
private int beating;
4848
private int failConnectCount = 0;
4949
private volatile bool isDisposed;
50+
private volatile bool shouldResetConnectionRetryCount;
5051
private long nonPreferredEndpointCount;
5152

5253
// private volatile int missedHeartbeats;
@@ -597,6 +598,8 @@ internal void OnHeartbeat(bool ifConnectedOnly)
597598
// We need to time that out and cleanup the PhysicalConnection if needed, otherwise that reader and socket will remain open
598599
// for the lifetime of the application due to being orphaned, yet still referenced by the active task doing the pipe read.
599600
case (int)State.ConnectedEstablished:
601+
// Track that we should reset the count on the next disconnect, but not do so in a loop
602+
shouldResetConnectionRetryCount = true;
600603
var tmp = physical;
601604
if (tmp != null)
602605
{
@@ -668,7 +671,14 @@ internal void OnHeartbeat(bool ifConnectedOnly)
668671
}
669672
break;
670673
case (int)State.Disconnected:
671-
Interlocked.Exchange(ref connectTimeoutRetryCount, 0);
674+
// Only if we should reset the connection count
675+
// This should only happen after a successful reconnection, and not every time we bounce from BeginConnectAsync -> Disconnected
676+
// in a failure loop case that happens when a node goes missing forever.
677+
if (shouldResetConnectionRetryCount)
678+
{
679+
shouldResetConnectionRetryCount = false;
680+
Interlocked.Exchange(ref connectTimeoutRetryCount, 0);
681+
}
672682
if (!ifConnectedOnly)
673683
{
674684
Multiplexer.Trace("Resurrecting " + ToString());

0 commit comments

Comments
 (0)