From b145338f1cd5050cba7938f3f3368bd12b11744c Mon Sep 17 00:00:00 2001 From: martintmk <103487740+martintmk@users.noreply.github.com> Date: Wed, 3 Jan 2024 10:03:10 +0100 Subject: [PATCH] Fix stack trace growing for opened circuit breaker (#1878) --- .../Controller/CircuitStateController.cs | 22 +++++------ .../BrokenCircuitExceptionTests.cs | 1 + .../Controller/CircuitStateControllerTests.cs | 38 +++++++++++++++++++ 3 files changed, 49 insertions(+), 12 deletions(-) diff --git a/src/Polly.Core/CircuitBreaker/Controller/CircuitStateController.cs b/src/Polly.Core/CircuitBreaker/Controller/CircuitStateController.cs index a00907ed227..e2333370884 100644 --- a/src/Polly.Core/CircuitBreaker/Controller/CircuitStateController.cs +++ b/src/Polly.Core/CircuitBreaker/Controller/CircuitStateController.cs @@ -20,7 +20,7 @@ internal sealed class CircuitStateController : IDisposable private DateTimeOffset _blockedUntil; private CircuitState _circuitState = CircuitState.Closed; private Outcome? _lastOutcome; - private BrokenCircuitException _breakingException = new(); + private Exception? _breakingException; private bool _disposed; #pragma warning disable S107 @@ -139,8 +139,8 @@ public ValueTask CloseCircuitAsync(ResilienceContext context) exception = _circuitState switch { - CircuitState.Open => _breakingException, - CircuitState.HalfOpen when !isHalfOpen => _breakingException, + CircuitState.Open => CreateBrokenCircuitException(), + CircuitState.HalfOpen when !isHalfOpen => CreateBrokenCircuitException(), CircuitState.Isolated => new IsolatedCircuitException(), _ => null }; @@ -302,17 +302,15 @@ private bool PermitHalfOpenCircuitTest_NeedsLock() private void SetLastHandledOutcome_NeedsLock(Outcome outcome) { _lastOutcome = outcome; - - if (outcome.Exception is Exception exception) - { - _breakingException = new BrokenCircuitException(BrokenCircuitException.DefaultMessage, exception); - } - else - { - _breakingException = new BrokenCircuitException(BrokenCircuitException.DefaultMessage); - } + _breakingException = outcome.Exception; } + private BrokenCircuitException CreateBrokenCircuitException() => _breakingException switch + { + Exception exception => new BrokenCircuitException(BrokenCircuitException.DefaultMessage, exception), + _ => new BrokenCircuitException(BrokenCircuitException.DefaultMessage) + }; + private void OpenCircuit_NeedsLock(Outcome outcome, bool manual, ResilienceContext context, out Task? scheduledTask) { OpenCircuitFor_NeedsLock(outcome, _breakDuration, manual, context, out scheduledTask); diff --git a/test/Polly.Core.Tests/CircuitBreaker/BrokenCircuitExceptionTests.cs b/test/Polly.Core.Tests/CircuitBreaker/BrokenCircuitExceptionTests.cs index 3fb989e7a59..7b69c987058 100644 --- a/test/Polly.Core.Tests/CircuitBreaker/BrokenCircuitExceptionTests.cs +++ b/test/Polly.Core.Tests/CircuitBreaker/BrokenCircuitExceptionTests.cs @@ -7,6 +7,7 @@ public class BrokenCircuitExceptionTests [Fact] public void Ctor_Ok() { + new BrokenCircuitException().Message.Should().Be("The circuit is now open and is not allowing calls."); new BrokenCircuitException("Dummy.").Message.Should().Be("Dummy."); new BrokenCircuitException("Dummy.", new InvalidOperationException()).Message.Should().Be("Dummy."); new BrokenCircuitException("Dummy.", new InvalidOperationException()).InnerException.Should().BeOfType(); diff --git a/test/Polly.Core.Tests/CircuitBreaker/Controller/CircuitStateControllerTests.cs b/test/Polly.Core.Tests/CircuitBreaker/Controller/CircuitStateControllerTests.cs index f69f4ca4600..c7b086b0780 100644 --- a/test/Polly.Core.Tests/CircuitBreaker/Controller/CircuitStateControllerTests.cs +++ b/test/Polly.Core.Tests/CircuitBreaker/Controller/CircuitStateControllerTests.cs @@ -124,6 +124,44 @@ public async Task OnActionPreExecute_CircuitOpenedByValue() GetBlockedTill(controller).Should().Be(_timeProvider.GetUtcNow() + _options.BreakDuration); } + [InlineData(true)] + [InlineData(false)] + [Theory] + public async Task OnActionPreExecute_CircuitOpened_EnsureExceptionStackTraceDoesNotGrow(bool innerException) + { + var stacks = new List(); + var context = ResilienceContextPool.Shared.Get(); + using var controller = CreateController(); + + await OpenCircuit( + controller, + innerException ? Outcome.FromException(new InvalidOperationException()) : Outcome.FromResult(99)); + + for (int i = 0; i < 100; i++) + { + try + { + (await controller.OnActionPreExecuteAsync(context)).Value.ThrowIfException(); + } + catch (BrokenCircuitException e) + { + stacks.Add(e.StackTrace!); + e.Message.Should().Be("The circuit is now open and is not allowing calls."); + + if (innerException) + { + e.InnerException.Should().BeOfType(); + } + else + { + e.InnerException.Should().BeNull(); + } + } + } + + stacks.Distinct().Should().HaveCount(1); + } + [Fact] public async Task HalfOpen_EnsureBreakDuration() {