Skip to content

Commit

Permalink
add __init__ tests
Browse files Browse the repository at this point in the history
  • Loading branch information
lobsterkatie committed Jul 23, 2024
1 parent da606f7 commit 7fc3a3d
Showing 1 changed file with 121 additions and 0 deletions.
121 changes: 121 additions & 0 deletions tests/sentry/utils/test_circuit_breaker2.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,121 @@
from unittest import TestCase
from unittest.mock import ANY, MagicMock, patch

from django.conf import settings
from redis.client import Pipeline

from sentry.ratelimits.sliding_windows import Quota, RedisSlidingWindowRateLimiter
from sentry.testutils.helpers.datetime import freeze_time
from sentry.utils.circuit_breaker2 import CircuitBreaker, CircuitBreakerConfig

# Note: These need to be relatively big. If the limit is too low, the RECOVERY quota isn't big
# enough to be useful, and if the window is too short, redis (which doesn't seem to listen to the
# @freezetime decorator) will expire the state keys.
DEFAULT_CONFIG: CircuitBreakerConfig = {
"error_limit": 200,
"error_limit_window": 3600, # 1 hr
"broken_state_duration": 120, # 2 min
}


@freeze_time()
class CircuitBreakerTest(TestCase):
def setUp(self) -> None:
self.config = DEFAULT_CONFIG
self.breaker = CircuitBreaker("dogs_are_great", self.config)

# Clear all existing keys from redis
self.breaker.redis_pipeline.flushall()
self.breaker.redis_pipeline.execute()

def test_sets_default_values(self):
breaker = self.breaker

assert breaker.__dict__ == {
"key": "dogs_are_great",
"broken_state_key": "dogs_are_great.circuit_breaker.broken",
"recovery_state_key": "dogs_are_great.circuit_breaker.in_recovery",
"error_limit": 200,
"recovery_error_limit": 20,
"window": 3600,
"window_granularity": 180,
"broken_state_duration": 120,
"recovery_duration": 7200,
# These can't be compared with a simple equality check and therefore are tested
# individually below
"limiter": ANY,
"primary_quota": ANY,
"recovery_quota": ANY,
"redis_pipeline": ANY,
}
assert isinstance(breaker.limiter, RedisSlidingWindowRateLimiter)
assert isinstance(breaker.primary_quota, Quota)
assert isinstance(breaker.recovery_quota, Quota)
assert breaker.primary_quota.__dict__ == {
"window_seconds": 3600,
"granularity_seconds": 180,
"limit": 200,
"prefix_override": "dogs_are_great.circuit_breaker.ok",
}
assert breaker.recovery_quota.__dict__ == {
"window_seconds": 3600,
"granularity_seconds": 180,
"limit": 20,
"prefix_override": "dogs_are_great.circuit_breaker.recovery",
}
assert isinstance(breaker.redis_pipeline, Pipeline)

@patch("sentry.utils.circuit_breaker2.logger")
def test_fixes_too_loose_recovery_limit(self, mock_logger: MagicMock):
config: CircuitBreakerConfig = {
**DEFAULT_CONFIG,
"error_limit": 200,
"recovery_error_limit": 400,
}

for settings_debug_value, expected_log_function in [
(True, mock_logger.error),
(False, mock_logger.warning),
]:
settings.DEBUG = settings_debug_value
breaker = CircuitBreaker("dogs_are_great", config)

expected_log_function.assert_called_with(
"Circuit breaker '%s' has a recovery error limit (%d) greater than or equal"
+ " to its primary error limit (%d). Using the stricter error-limit-based"
+ " default (%d) instead.",
breaker.key,
400,
200,
20,
)
assert breaker.recovery_error_limit == 20

@patch("sentry.utils.circuit_breaker2.logger")
def test_fixes_mismatched_state_durations(self, mock_logger: MagicMock):
config: CircuitBreakerConfig = {
**DEFAULT_CONFIG,
"error_limit_window": 600,
"broken_state_duration": 100,
"recovery_duration": 200,
}
for settings_debug_value, expected_log_function in [
(True, mock_logger.error),
(False, mock_logger.warning),
]:
settings.DEBUG = settings_debug_value
breaker = CircuitBreaker("dogs_are_great", config)

expected_log_function.assert_called_with(
"Circuit breaker '%s' has BROKEN and RECOVERY state durations (%d and %d sec, respectively)"
+ " which together are less than the main error limit window (%d sec). This can lead to the"
+ " breaker getting tripped unexpectedly, until the original spike in errors clears the"
+ " main time window. Extending RECOVERY period to %d seconds, to give the primary quota time"
+ " to clear.",
breaker.key,
100,
200,
600,
500,
)
assert breaker.recovery_duration == 500

0 comments on commit 7fc3a3d

Please sign in to comment.