From fdfc30c287354f2d0551915fd048f3c8b7134adb Mon Sep 17 00:00:00 2001 From: vladvildanov Date: Wed, 22 Oct 2025 14:55:02 +0300 Subject: [PATCH] Refactor healthcheck to use PING instead of ECHO --- redis/asyncio/multidb/client.py | 2 ++ redis/asyncio/multidb/config.py | 4 ++-- redis/asyncio/multidb/healthcheck.py | 15 ++++---------- redis/multidb/client.py | 2 ++ redis/multidb/config.py | 4 ++-- redis/multidb/healthcheck.py | 15 ++++---------- redis/utils.py | 20 +++++++++++++++++++ .../test_asyncio/test_multidb/test_config.py | 4 ++-- .../test_multidb/test_healthcheck.py | 16 +++++++-------- tests/test_multidb/test_config.py | 4 ++-- tests/test_multidb/test_healthcheck.py | 16 +++++++-------- 11 files changed, 56 insertions(+), 46 deletions(-) diff --git a/redis/asyncio/multidb/client.py b/redis/asyncio/multidb/client.py index 6bea588196..d3e3f241d2 100644 --- a/redis/asyncio/multidb/client.py +++ b/redis/asyncio/multidb/client.py @@ -14,10 +14,12 @@ from redis.multidb.circuit import State as CBState from redis.multidb.exception import NoValidDatabaseException, UnhealthyDatabaseException from redis.typing import ChannelT, EncodableT, KeyT +from redis.utils import experimental logger = logging.getLogger(__name__) +@experimental class MultiDBClient(AsyncRedisModuleCommands, AsyncCoreCommands): """ Client that operates on multiple logical Redis databases. diff --git a/redis/asyncio/multidb/config.py b/redis/asyncio/multidb/config.py index 71f69ad133..cd0675baeb 100644 --- a/redis/asyncio/multidb/config.py +++ b/redis/asyncio/multidb/config.py @@ -20,9 +20,9 @@ DEFAULT_HEALTH_CHECK_INTERVAL, DEFAULT_HEALTH_CHECK_POLICY, DEFAULT_HEALTH_CHECK_PROBES, - EchoHealthCheck, HealthCheck, HealthCheckPolicies, + PingHealthCheck, ) from redis.asyncio.retry import Retry from redis.backoff import ExponentialWithJitterBackoff, NoBackoff @@ -203,7 +203,7 @@ def default_failure_detectors(self) -> List[AsyncFailureDetector]: def default_health_checks(self) -> List[HealthCheck]: return [ - EchoHealthCheck(), + PingHealthCheck(), ] def default_failover_strategy(self) -> AsyncFailoverStrategy: diff --git a/redis/asyncio/multidb/healthcheck.py b/redis/asyncio/multidb/healthcheck.py index dcb787f6ed..d2269f707d 100644 --- a/redis/asyncio/multidb/healthcheck.py +++ b/redis/asyncio/multidb/healthcheck.py @@ -170,26 +170,19 @@ class HealthCheckPolicies(Enum): DEFAULT_HEALTH_CHECK_POLICY: HealthCheckPolicies = HealthCheckPolicies.HEALTHY_ALL -class EchoHealthCheck(HealthCheck): +class PingHealthCheck(HealthCheck): """ - Health check based on ECHO command. + Health check based on PING command. """ async def check_health(self, database) -> bool: - expected_message = ["healthcheck", b"healthcheck"] - if isinstance(database.client, Redis): - actual_message = await database.client.execute_command( - "ECHO", "healthcheck" - ) - return actual_message in expected_message + return await database.client.execute_command("PING") else: # For a cluster checks if all nodes are healthy. all_nodes = database.client.get_nodes() for node in all_nodes: - actual_message = await node.execute_command("ECHO", "healthcheck") - - if actual_message not in expected_message: + if not await node.redis_connection.execute_command("PING"): return False return True diff --git a/redis/multidb/client.py b/redis/multidb/client.py index 02c3516eee..c46a53af32 100644 --- a/redis/multidb/client.py +++ b/redis/multidb/client.py @@ -15,10 +15,12 @@ from redis.multidb.exception import NoValidDatabaseException, UnhealthyDatabaseException from redis.multidb.failure_detector import FailureDetector from redis.multidb.healthcheck import HealthCheck, HealthCheckPolicy +from redis.utils import experimental logger = logging.getLogger(__name__) +@experimental class MultiDBClient(RedisModuleCommands, CoreCommands): """ Client that operates on multiple logical Redis databases. diff --git a/redis/multidb/config.py b/redis/multidb/config.py index 8ce960c2dc..b3187c79fb 100644 --- a/redis/multidb/config.py +++ b/redis/multidb/config.py @@ -32,9 +32,9 @@ DEFAULT_HEALTH_CHECK_INTERVAL, DEFAULT_HEALTH_CHECK_POLICY, DEFAULT_HEALTH_CHECK_PROBES, - EchoHealthCheck, HealthCheck, HealthCheckPolicies, + PingHealthCheck, ) from redis.retry import Retry @@ -200,7 +200,7 @@ def default_failure_detectors(self) -> List[FailureDetector]: def default_health_checks(self) -> List[HealthCheck]: return [ - EchoHealthCheck(), + PingHealthCheck(), ] def default_failover_strategy(self) -> FailoverStrategy: diff --git a/redis/multidb/healthcheck.py b/redis/multidb/healthcheck.py index 5deda82f24..43440d5518 100644 --- a/redis/multidb/healthcheck.py +++ b/redis/multidb/healthcheck.py @@ -169,26 +169,19 @@ class HealthCheckPolicies(Enum): DEFAULT_HEALTH_CHECK_POLICY: HealthCheckPolicies = HealthCheckPolicies.HEALTHY_ALL -class EchoHealthCheck(HealthCheck): +class PingHealthCheck(HealthCheck): """ - Health check based on ECHO command. + Health check based on PING command. """ def check_health(self, database) -> bool: - expected_message = ["healthcheck", b"healthcheck"] - if isinstance(database.client, Redis): - actual_message = database.client.execute_command("ECHO", "healthcheck") - return actual_message in expected_message + return database.client.execute_command("PING") else: # For a cluster checks if all nodes are healthy. all_nodes = database.client.get_nodes() for node in all_nodes: - actual_message = node.redis_connection.execute_command( - "ECHO", "healthcheck" - ) - - if actual_message not in expected_message: + if not node.redis_connection.execute_command("PING"): return False return True diff --git a/redis/utils.py b/redis/utils.py index 5ae8fb25fc..66c1480a9b 100644 --- a/redis/utils.py +++ b/redis/utils.py @@ -1,6 +1,7 @@ import datetime import logging import textwrap +import warnings from collections.abc import Callable from contextlib import contextmanager from functools import wraps @@ -326,3 +327,22 @@ async def dummy_fail_async(): Async fake function for a Retry object if you don't need to handle each failure. """ pass + + +def experimental(cls): + """ + Decorator to mark a class as experimental. + """ + original_init = cls.__init__ + + @wraps(original_init) + def new_init(self, *args, **kwargs): + warnings.warn( + f"{cls.__name__} is an experimental and may change or be removed in future versions.", + category=UserWarning, + stacklevel=2, + ) + original_init(self, *args, **kwargs) + + cls.__init__ = new_init + return cls diff --git a/tests/test_asyncio/test_multidb/test_config.py b/tests/test_asyncio/test_multidb/test_config.py index d05c7a8a12..5ffeb97e06 100644 --- a/tests/test_asyncio/test_multidb/test_config.py +++ b/tests/test_asyncio/test_multidb/test_config.py @@ -19,7 +19,7 @@ FailureDetectorAsyncWrapper, AsyncFailureDetector, ) -from redis.asyncio.multidb.healthcheck import EchoHealthCheck, HealthCheck +from redis.asyncio.multidb.healthcheck import PingHealthCheck, HealthCheck from redis.asyncio.retry import Retry from redis.multidb.circuit import CircuitBreaker @@ -58,7 +58,7 @@ def test_default_config(self): config.default_failure_detectors()[0], FailureDetectorAsyncWrapper ) assert len(config.default_health_checks()) == 1 - assert isinstance(config.default_health_checks()[0], EchoHealthCheck) + assert isinstance(config.default_health_checks()[0], PingHealthCheck) assert config.health_check_interval == DEFAULT_HEALTH_CHECK_INTERVAL assert isinstance( config.default_failover_strategy(), WeightBasedFailoverStrategy diff --git a/tests/test_asyncio/test_multidb/test_healthcheck.py b/tests/test_asyncio/test_multidb/test_healthcheck.py index 3e7ac42cd9..5a08f21a68 100644 --- a/tests/test_asyncio/test_multidb/test_healthcheck.py +++ b/tests/test_asyncio/test_multidb/test_healthcheck.py @@ -3,7 +3,7 @@ from redis.asyncio.multidb.database import Database from redis.asyncio.multidb.healthcheck import ( - EchoHealthCheck, + PingHealthCheck, LagAwareHealthCheck, HealthCheck, HealthyAllPolicy, @@ -208,15 +208,15 @@ async def test_policy_raise_unhealthy_database_exception_if_exception_occurs_on_ @pytest.mark.onlynoncluster -class TestEchoHealthCheck: +class TestPingHealthCheck: @pytest.mark.asyncio async def test_database_is_healthy_on_echo_response(self, mock_client, mock_cb): """ Mocking responses to mix error and actual responses to ensure that health check retry according to given configuration. """ - mock_client.execute_command = AsyncMock(side_effect=["healthcheck"]) - hc = EchoHealthCheck() + mock_client.execute_command = AsyncMock(side_effect=["PONG"]) + hc = PingHealthCheck() db = Database(mock_client, mock_cb, 0.9) assert await hc.check_health(db) @@ -230,8 +230,8 @@ async def test_database_is_unhealthy_on_incorrect_echo_response( Mocking responses to mix error and actual responses to ensure that health check retry according to given configuration. """ - mock_client.execute_command = AsyncMock(side_effect=["wrong"]) - hc = EchoHealthCheck() + mock_client.execute_command = AsyncMock(side_effect=[False]) + hc = PingHealthCheck() db = Database(mock_client, mock_cb, 0.9) assert not await hc.check_health(db) @@ -241,9 +241,9 @@ async def test_database_is_unhealthy_on_incorrect_echo_response( async def test_database_close_circuit_on_successful_healthcheck( self, mock_client, mock_cb ): - mock_client.execute_command = AsyncMock(side_effect=["healthcheck"]) + mock_client.execute_command = AsyncMock(side_effect=["PONG"]) mock_cb.state = CBState.HALF_OPEN - hc = EchoHealthCheck() + hc = PingHealthCheck() db = Database(mock_client, mock_cb, 0.9) assert await hc.check_health(db) diff --git a/tests/test_multidb/test_config.py b/tests/test_multidb/test_config.py index 351f789971..0683e9e9f6 100644 --- a/tests/test_multidb/test_config.py +++ b/tests/test_multidb/test_config.py @@ -16,7 +16,7 @@ ) from redis.multidb.database import Database from redis.multidb.failure_detector import CommandFailureDetector, FailureDetector -from redis.multidb.healthcheck import EchoHealthCheck, HealthCheck +from redis.multidb.healthcheck import PingHealthCheck, HealthCheck from redis.multidb.failover import WeightBasedFailoverStrategy, FailoverStrategy from redis.retry import Retry @@ -53,7 +53,7 @@ def test_default_config(self): assert len(config.default_failure_detectors()) == 1 assert isinstance(config.default_failure_detectors()[0], CommandFailureDetector) assert len(config.default_health_checks()) == 1 - assert isinstance(config.default_health_checks()[0], EchoHealthCheck) + assert isinstance(config.default_health_checks()[0], PingHealthCheck) assert config.health_check_interval == DEFAULT_HEALTH_CHECK_INTERVAL assert isinstance( config.default_failover_strategy(), WeightBasedFailoverStrategy diff --git a/tests/test_multidb/test_healthcheck.py b/tests/test_multidb/test_healthcheck.py index fb1f1e4148..b70da06091 100644 --- a/tests/test_multidb/test_healthcheck.py +++ b/tests/test_multidb/test_healthcheck.py @@ -5,7 +5,7 @@ from redis.multidb.database import Database from redis.http.http_client import HttpError from redis.multidb.healthcheck import ( - EchoHealthCheck, + PingHealthCheck, LagAwareHealthCheck, HealthCheck, HealthyAllPolicy, @@ -201,14 +201,14 @@ def test_policy_raise_unhealthy_database_exception_if_exception_occurs_on_failed @pytest.mark.onlynoncluster -class TestEchoHealthCheck: +class TestPingHealthCheck: def test_database_is_healthy_on_echo_response(self, mock_client, mock_cb): """ Mocking responses to mix error and actual responses to ensure that health check retry according to given configuration. """ - mock_client.execute_command.return_value = "healthcheck" - hc = EchoHealthCheck() + mock_client.execute_command.return_value = "PONG" + hc = PingHealthCheck() db = Database(mock_client, mock_cb, 0.9) assert hc.check_health(db) @@ -221,8 +221,8 @@ def test_database_is_unhealthy_on_incorrect_echo_response( Mocking responses to mix error and actual responses to ensure that health check retry according to given configuration. """ - mock_client.execute_command.return_value = "wrong" - hc = EchoHealthCheck() + mock_client.execute_command.return_value = False + hc = PingHealthCheck() db = Database(mock_client, mock_cb, 0.9) assert not hc.check_health(db) @@ -231,9 +231,9 @@ def test_database_is_unhealthy_on_incorrect_echo_response( def test_database_close_circuit_on_successful_healthcheck( self, mock_client, mock_cb ): - mock_client.execute_command.return_value = "healthcheck" + mock_client.execute_command.return_value = "PONG" mock_cb.state = CBState.HALF_OPEN - hc = EchoHealthCheck() + hc = PingHealthCheck() db = Database(mock_client, mock_cb, 0.9) assert hc.check_health(db)