Skip to content

Commit 7bfd4c5

Browse files
authoredMar 26, 2025··
Fixed infinitely recursive health checks. (#3557)
1 parent c42df94 commit 7bfd4c5

File tree

2 files changed

+61
-15
lines changed

2 files changed

+61
-15
lines changed
 

‎redis/asyncio/connection.py

+31-8
Original file line numberDiff line numberDiff line change
@@ -293,6 +293,9 @@ def set_parser(self, parser_class: Type[BaseParser]) -> None:
293293

294294
async def connect(self):
295295
"""Connects to the Redis server if not already connected"""
296+
await self.connect_check_health(check_health=True)
297+
298+
async def connect_check_health(self, check_health: bool = True):
296299
if self.is_connected:
297300
return
298301
try:
@@ -311,7 +314,7 @@ async def connect(self):
311314
try:
312315
if not self.redis_connect_func:
313316
# Use the default on_connect function
314-
await self.on_connect()
317+
await self.on_connect_check_health(check_health=check_health)
315318
else:
316319
# Use the passed function redis_connect_func
317320
(
@@ -350,6 +353,9 @@ def get_protocol(self):
350353

351354
async def on_connect(self) -> None:
352355
"""Initialize the connection, authenticate and select a database"""
356+
await self.on_connect_check_health(check_health=True)
357+
358+
async def on_connect_check_health(self, check_health: bool = True) -> None:
353359
self._parser.on_connect(self)
354360
parser = self._parser
355361

@@ -407,7 +413,7 @@ async def on_connect(self) -> None:
407413
# update cluster exception classes
408414
self._parser.EXCEPTION_CLASSES = parser.EXCEPTION_CLASSES
409415
self._parser.on_connect(self)
410-
await self.send_command("HELLO", self.protocol)
416+
await self.send_command("HELLO", self.protocol, check_health=check_health)
411417
response = await self.read_response()
412418
# if response.get(b"proto") != self.protocol and response.get(
413419
# "proto"
@@ -416,18 +422,35 @@ async def on_connect(self) -> None:
416422

417423
# if a client_name is given, set it
418424
if self.client_name:
419-
await self.send_command("CLIENT", "SETNAME", self.client_name)
425+
await self.send_command(
426+
"CLIENT",
427+
"SETNAME",
428+
self.client_name,
429+
check_health=check_health,
430+
)
420431
if str_if_bytes(await self.read_response()) != "OK":
421432
raise ConnectionError("Error setting client name")
422433

423434
# set the library name and version, pipeline for lower startup latency
424435
if self.lib_name:
425-
await self.send_command("CLIENT", "SETINFO", "LIB-NAME", self.lib_name)
436+
await self.send_command(
437+
"CLIENT",
438+
"SETINFO",
439+
"LIB-NAME",
440+
self.lib_name,
441+
check_health=check_health,
442+
)
426443
if self.lib_version:
427-
await self.send_command("CLIENT", "SETINFO", "LIB-VER", self.lib_version)
444+
await self.send_command(
445+
"CLIENT",
446+
"SETINFO",
447+
"LIB-VER",
448+
self.lib_version,
449+
check_health=check_health,
450+
)
428451
# if a database is specified, switch to it. Also pipeline this
429452
if self.db:
430-
await self.send_command("SELECT", self.db)
453+
await self.send_command("SELECT", self.db, check_health=check_health)
431454

432455
# read responses from pipeline
433456
for _ in (sent for sent in (self.lib_name, self.lib_version) if sent):
@@ -489,8 +512,8 @@ async def send_packed_command(
489512
self, command: Union[bytes, str, Iterable[bytes]], check_health: bool = True
490513
) -> None:
491514
if not self.is_connected:
492-
await self.connect()
493-
elif check_health:
515+
await self.connect_check_health(check_health=False)
516+
if check_health:
494517
await self.check_health()
495518

496519
try:

‎redis/connection.py

+30-7
Original file line numberDiff line numberDiff line change
@@ -376,6 +376,9 @@ def set_parser(self, parser_class):
376376

377377
def connect(self):
378378
"Connects to the Redis server if not already connected"
379+
self.connect_check_health(check_health=True)
380+
381+
def connect_check_health(self, check_health: bool = True):
379382
if self._sock:
380383
return
381384
try:
@@ -391,7 +394,7 @@ def connect(self):
391394
try:
392395
if self.redis_connect_func is None:
393396
# Use the default on_connect function
394-
self.on_connect()
397+
self.on_connect_check_health(check_health=check_health)
395398
else:
396399
# Use the passed function redis_connect_func
397400
self.redis_connect_func(self)
@@ -421,6 +424,9 @@ def _error_message(self, exception):
421424
return format_error_message(self._host_error(), exception)
422425

423426
def on_connect(self):
427+
self.on_connect_check_health(check_health=True)
428+
429+
def on_connect_check_health(self, check_health: bool = True):
424430
"Initialize the connection, authenticate and select a database"
425431
self._parser.on_connect(self)
426432
parser = self._parser
@@ -479,7 +485,7 @@ def on_connect(self):
479485
# update cluster exception classes
480486
self._parser.EXCEPTION_CLASSES = parser.EXCEPTION_CLASSES
481487
self._parser.on_connect(self)
482-
self.send_command("HELLO", self.protocol)
488+
self.send_command("HELLO", self.protocol, check_health=check_health)
483489
self.handshake_metadata = self.read_response()
484490
if (
485491
self.handshake_metadata.get(b"proto") != self.protocol
@@ -489,28 +495,45 @@ def on_connect(self):
489495

490496
# if a client_name is given, set it
491497
if self.client_name:
492-
self.send_command("CLIENT", "SETNAME", self.client_name)
498+
self.send_command(
499+
"CLIENT",
500+
"SETNAME",
501+
self.client_name,
502+
check_health=check_health,
503+
)
493504
if str_if_bytes(self.read_response()) != "OK":
494505
raise ConnectionError("Error setting client name")
495506

496507
try:
497508
# set the library name and version
498509
if self.lib_name:
499-
self.send_command("CLIENT", "SETINFO", "LIB-NAME", self.lib_name)
510+
self.send_command(
511+
"CLIENT",
512+
"SETINFO",
513+
"LIB-NAME",
514+
self.lib_name,
515+
check_health=check_health,
516+
)
500517
self.read_response()
501518
except ResponseError:
502519
pass
503520

504521
try:
505522
if self.lib_version:
506-
self.send_command("CLIENT", "SETINFO", "LIB-VER", self.lib_version)
523+
self.send_command(
524+
"CLIENT",
525+
"SETINFO",
526+
"LIB-VER",
527+
self.lib_version,
528+
check_health=check_health,
529+
)
507530
self.read_response()
508531
except ResponseError:
509532
pass
510533

511534
# if a database is specified, switch to it
512535
if self.db:
513-
self.send_command("SELECT", self.db)
536+
self.send_command("SELECT", self.db, check_health=check_health)
514537
if str_if_bytes(self.read_response()) != "OK":
515538
raise ConnectionError("Invalid Database")
516539

@@ -552,7 +575,7 @@ def check_health(self):
552575
def send_packed_command(self, command, check_health=True):
553576
"""Send an already packed command to the Redis server"""
554577
if not self._sock:
555-
self.connect()
578+
self.connect_check_health(check_health=False)
556579
# guard against health check recursion
557580
if check_health:
558581
self.check_health()

0 commit comments

Comments
 (0)
Please sign in to comment.