@@ -500,28 +500,37 @@ async def _disconnect_raise(self, conn: Connection, error: Exception):
500
500
):
501
501
raise error
502
502
503
- # COMMAND EXECUTION AND PROTOCOL PARSING
504
- async def execute_command (self , * args , ** options ):
505
- """Execute a command and return a parsed response"""
506
- await self .initialize ()
507
- pool = self .connection_pool
508
- command_name = args [0 ]
509
- conn = self .connection or await pool .get_connection (command_name , ** options )
510
-
511
- if self .single_connection_client :
512
- await self ._single_conn_lock .acquire ()
503
+ async def _try_send_command_parse_response (self , conn , * args , ** options ):
513
504
try :
514
505
return await conn .retry .call_with_retry (
515
506
lambda : self ._send_command_parse_response (
516
- conn , command_name , * args , ** options
507
+ conn , args [ 0 ] , * args , ** options
517
508
),
518
509
lambda error : self ._disconnect_raise (conn , error ),
519
510
)
511
+ except asyncio .CancelledError :
512
+ await conn .disconnect (nowait = True )
513
+ raise
520
514
finally :
521
515
if self .single_connection_client :
522
516
self ._single_conn_lock .release ()
523
517
if not self .connection :
524
- await pool .release (conn )
518
+ await self .connection_pool .release (conn )
519
+
520
+ # COMMAND EXECUTION AND PROTOCOL PARSING
521
+ async def execute_command (self , * args , ** options ):
522
+ """Execute a command and return a parsed response"""
523
+ await self .initialize ()
524
+ pool = self .connection_pool
525
+ command_name = args [0 ]
526
+ conn = self .connection or await pool .get_connection (command_name , ** options )
527
+
528
+ if self .single_connection_client :
529
+ await self ._single_conn_lock .acquire ()
530
+
531
+ return await asyncio .shield (
532
+ self ._try_send_command_parse_response (conn , * args , ** options )
533
+ )
525
534
526
535
async def parse_response (
527
536
self , connection : Connection , command_name : Union [str , bytes ], ** options
@@ -702,6 +711,11 @@ async def reset(self):
702
711
self .pending_unsubscribe_patterns = set ()
703
712
704
713
def close (self ) -> Awaitable [NoReturn ]:
714
+ # In case a connection property does not yet exist
715
+ # (due to a crash earlier in the Redis() constructor), return
716
+ # immediately as there is nothing to clean-up.
717
+ if not hasattr (self , "connection" ):
718
+ return
705
719
return self .reset ()
706
720
707
721
async def on_connect (self , connection : Connection ):
@@ -760,10 +774,18 @@ async def _disconnect_raise_connect(self, conn, error):
760
774
is not a TimeoutError. Otherwise, try to reconnect
761
775
"""
762
776
await conn .disconnect ()
777
+
763
778
if not (conn .retry_on_timeout and isinstance (error , TimeoutError )):
764
779
raise error
765
780
await conn .connect ()
766
781
782
+ async def _try_execute (self , conn , command , * arg , ** kwargs ):
783
+ try :
784
+ return await command (* arg , ** kwargs )
785
+ except asyncio .CancelledError :
786
+ await conn .disconnect ()
787
+ raise
788
+
767
789
async def _execute (self , conn , command , * args , ** kwargs ):
768
790
"""
769
791
Connect manually upon disconnection. If the Redis server is down,
@@ -772,9 +794,11 @@ async def _execute(self, conn, command, *args, **kwargs):
772
794
called by the # connection to resubscribe us to any channels and
773
795
patterns we were previously listening to
774
796
"""
775
- return await conn .retry .call_with_retry (
776
- lambda : command (* args , ** kwargs ),
777
- lambda error : self ._disconnect_raise_connect (conn , error ),
797
+ return await asyncio .shield (
798
+ conn .retry .call_with_retry (
799
+ lambda : self ._try_execute (conn , command , * args , ** kwargs ),
800
+ lambda error : self ._disconnect_raise_connect (conn , error ),
801
+ )
778
802
)
779
803
780
804
async def parse_response (self , block : bool = True , timeout : float = 0 ):
@@ -1178,6 +1202,18 @@ async def _disconnect_reset_raise(self, conn, error):
1178
1202
await self .reset ()
1179
1203
raise
1180
1204
1205
+ async def _try_send_command_parse_response (self , conn , * args , ** options ):
1206
+ try :
1207
+ return await conn .retry .call_with_retry (
1208
+ lambda : self ._send_command_parse_response (
1209
+ conn , args [0 ], * args , ** options
1210
+ ),
1211
+ lambda error : self ._disconnect_reset_raise (conn , error ),
1212
+ )
1213
+ except asyncio .CancelledError :
1214
+ await conn .disconnect ()
1215
+ raise
1216
+
1181
1217
async def immediate_execute_command (self , * args , ** options ):
1182
1218
"""
1183
1219
Execute a command immediately, but don't auto-retry on a
@@ -1193,12 +1229,8 @@ async def immediate_execute_command(self, *args, **options):
1193
1229
command_name , self .shard_hint
1194
1230
)
1195
1231
self .connection = conn
1196
-
1197
- return await conn .retry .call_with_retry (
1198
- lambda : self ._send_command_parse_response (
1199
- conn , command_name , * args , ** options
1200
- ),
1201
- lambda error : self ._disconnect_reset_raise (conn , error ),
1232
+ return await asyncio .shield (
1233
+ self ._try_send_command_parse_response (conn , * args , ** options )
1202
1234
)
1203
1235
1204
1236
def pipeline_execute_command (self , * args , ** options ):
@@ -1366,6 +1398,19 @@ async def _disconnect_raise_reset(self, conn: Connection, error: Exception):
1366
1398
await self .reset ()
1367
1399
raise
1368
1400
1401
+ async def _try_execute (self , conn , execute , stack , raise_on_error ):
1402
+ try :
1403
+ return await conn .retry .call_with_retry (
1404
+ lambda : execute (conn , stack , raise_on_error ),
1405
+ lambda error : self ._disconnect_raise_reset (conn , error ),
1406
+ )
1407
+ except asyncio .CancelledError :
1408
+ # not supposed to be possible, yet here we are
1409
+ await conn .disconnect (nowait = True )
1410
+ raise
1411
+ finally :
1412
+ await self .reset ()
1413
+
1369
1414
async def execute (self , raise_on_error : bool = True ):
1370
1415
"""Execute all the commands in the current pipeline"""
1371
1416
stack = self .command_stack
@@ -1387,10 +1432,11 @@ async def execute(self, raise_on_error: bool = True):
1387
1432
conn = cast (Connection , conn )
1388
1433
1389
1434
try :
1390
- return await conn .retry .call_with_retry (
1391
- lambda : execute (conn , stack , raise_on_error ),
1392
- lambda error : self ._disconnect_raise_reset (conn , error ),
1435
+ return await asyncio .shield (
1436
+ self ._try_execute (conn , execute , stack , raise_on_error )
1393
1437
)
1438
+ except RuntimeError :
1439
+ await self .reset ()
1394
1440
finally :
1395
1441
await self .reset ()
1396
1442
0 commit comments