-
Notifications
You must be signed in to change notification settings - Fork 2.6k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
feat: Implemented redis.asyncio.SentinelBlockingConnectionPool. #3321
base: master
Are you sure you want to change the base?
Changes from all commits
367d65d
582cfde
9b1fbd6
48bf87c
f7c7a87
7b3c986
8cbb03a
1bd3cd7
9e052a8
3f7daba
6a99b49
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,10 +1,20 @@ | ||
import asyncio | ||
import random | ||
import weakref | ||
from typing import AsyncIterator, Iterable, Mapping, Optional, Sequence, Tuple, Type | ||
from typing import ( | ||
AsyncIterator, | ||
Iterable, | ||
Mapping, | ||
Optional, | ||
Sequence, | ||
Tuple, | ||
Type, | ||
Union, | ||
) | ||
|
||
from redis.asyncio.client import Redis | ||
from redis.asyncio.connection import ( | ||
BlockingConnectionPool, | ||
Connection, | ||
ConnectionPool, | ||
EncodableT, | ||
|
@@ -97,6 +107,55 @@ class SentinelManagedSSLConnection(SentinelManagedConnection, SSLConnection): | |
pass | ||
|
||
|
||
class SentinelConnectionPoolProxy: | ||
def __init__( | ||
self, | ||
connection_pool, | ||
is_master, | ||
check_connection, | ||
service_name, | ||
sentinel_manager, | ||
): | ||
self.connection_pool_ref = weakref.ref(connection_pool) | ||
self.is_master = is_master | ||
self.check_connection = check_connection | ||
self.service_name = service_name | ||
self.sentinel_manager = sentinel_manager | ||
self.reset() | ||
|
||
def reset(self): | ||
self.master_address = None | ||
self.slave_rr_counter = None | ||
|
||
async def get_master_address(self): | ||
master_address = await self.sentinel_manager.discover_master(self.service_name) | ||
if self.is_master and self.master_address != master_address: | ||
self.master_address = master_address | ||
# disconnect any idle connections so that they reconnect | ||
# to the new master the next time that they are used. | ||
connection_pool = self.connection_pool_ref() | ||
if connection_pool is not None: | ||
await connection_pool.disconnect(inuse_connections=False) | ||
return master_address | ||
|
||
async def rotate_slaves(self) -> AsyncIterator: | ||
"""Round-robin slave balancer""" | ||
slaves = await self.sentinel_manager.discover_slaves(self.service_name) | ||
if slaves: | ||
if self.slave_rr_counter is None: | ||
self.slave_rr_counter = random.randint(0, len(slaves) - 1) | ||
for _ in range(len(slaves)): | ||
self.slave_rr_counter = (self.slave_rr_counter + 1) % len(slaves) | ||
slave = slaves[self.slave_rr_counter] | ||
yield slave | ||
# Fallback to the master connection | ||
try: | ||
yield await self.get_master_address() | ||
except MasterNotFoundError: | ||
pass | ||
raise SlaveNotFoundError(f"No slave found for {self.service_name!r}") | ||
|
||
|
||
class SentinelConnectionPool(ConnectionPool): | ||
""" | ||
Sentinel backed connection pool. | ||
|
@@ -116,12 +175,17 @@ def __init__(self, service_name, sentinel_manager, **kwargs): | |
) | ||
self.is_master = kwargs.pop("is_master", True) | ||
self.check_connection = kwargs.pop("check_connection", False) | ||
self.proxy = SentinelConnectionPoolProxy( | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Well, conceptually proxies are used to add an additional functionality before or after we want to access the original object. But here proxy are used just to override super class methods that breaks the functionality of child. Proxy here is the workaround, to hide a problem with inheritance that we have. We have to either fix it in ConnectionPool so it can be inheritable, or doesn't inherit this at all |
||
connection_pool=self, | ||
is_master=self.is_master, | ||
check_connection=self.check_connection, | ||
service_name=service_name, | ||
sentinel_manager=sentinel_manager, | ||
) | ||
super().__init__(**kwargs) | ||
self.connection_kwargs["connection_pool"] = weakref.proxy(self) | ||
self.connection_kwargs["connection_pool"] = self.proxy | ||
self.service_name = service_name | ||
self.sentinel_manager = sentinel_manager | ||
self.master_address = None | ||
self.slave_rr_counter = None | ||
|
||
def __repr__(self): | ||
return ( | ||
|
@@ -131,8 +195,11 @@ def __repr__(self): | |
|
||
def reset(self): | ||
super().reset() | ||
self.master_address = None | ||
self.slave_rr_counter = None | ||
self.proxy.reset() | ||
|
||
@property | ||
def master_address(self): | ||
return self.proxy.master_address | ||
|
||
def owns_connection(self, connection: Connection): | ||
check = not self.is_master or ( | ||
|
@@ -141,31 +208,70 @@ def owns_connection(self, connection: Connection): | |
return check and super().owns_connection(connection) | ||
|
||
async def get_master_address(self): | ||
master_address = await self.sentinel_manager.discover_master(self.service_name) | ||
if self.is_master: | ||
if self.master_address != master_address: | ||
self.master_address = master_address | ||
# disconnect any idle connections so that they reconnect | ||
# to the new master the next time that they are used. | ||
await self.disconnect(inuse_connections=False) | ||
return master_address | ||
return await self.proxy.get_master_address() | ||
|
||
async def rotate_slaves(self) -> AsyncIterator: | ||
def rotate_slaves(self) -> AsyncIterator: | ||
"""Round-robin slave balancer""" | ||
slaves = await self.sentinel_manager.discover_slaves(self.service_name) | ||
if slaves: | ||
if self.slave_rr_counter is None: | ||
self.slave_rr_counter = random.randint(0, len(slaves) - 1) | ||
for _ in range(len(slaves)): | ||
self.slave_rr_counter = (self.slave_rr_counter + 1) % len(slaves) | ||
slave = slaves[self.slave_rr_counter] | ||
yield slave | ||
# Fallback to the master connection | ||
try: | ||
yield await self.get_master_address() | ||
except MasterNotFoundError: | ||
pass | ||
raise SlaveNotFoundError(f"No slave found for {self.service_name!r}") | ||
return self.proxy.rotate_slaves() | ||
|
||
|
||
class SentinelBlockingConnectionPool(BlockingConnectionPool): | ||
vladvildanov marked this conversation as resolved.
Show resolved
Hide resolved
|
||
""" | ||
Sentinel blocking connection pool. | ||
If ``check_connection`` flag is set to True, SentinelManagedConnection | ||
sends a PING command right after establishing the connection. | ||
""" | ||
|
||
def __init__(self, service_name, sentinel_manager, **kwargs): | ||
kwargs["connection_class"] = kwargs.get( | ||
"connection_class", | ||
( | ||
SentinelManagedSSLConnection | ||
if kwargs.pop("ssl", False) | ||
else SentinelManagedConnection | ||
), | ||
) | ||
self.is_master = kwargs.pop("is_master", True) | ||
self.check_connection = kwargs.pop("check_connection", False) | ||
self.proxy = SentinelConnectionPoolProxy( | ||
connection_pool=self, | ||
is_master=self.is_master, | ||
check_connection=self.check_connection, | ||
service_name=service_name, | ||
sentinel_manager=sentinel_manager, | ||
) | ||
super().__init__(**kwargs) | ||
self.connection_kwargs["connection_pool"] = self.proxy | ||
self.service_name = service_name | ||
self.sentinel_manager = sentinel_manager | ||
|
||
def __repr__(self): | ||
return ( | ||
f"<{self.__class__.__module__}.{self.__class__.__name__}" | ||
f"(service={self.service_name}({self.is_master and 'master' or 'slave'}))>" | ||
) | ||
|
||
def reset(self): | ||
super().reset() | ||
self.proxy.reset() | ||
|
||
@property | ||
def master_address(self): | ||
return self.proxy.master_address | ||
|
||
def owns_connection(self, connection: Connection): | ||
check = not self.is_master or ( | ||
self.is_master and self.master_address == (connection.host, connection.port) | ||
) | ||
return check and super().owns_connection(connection) | ||
|
||
async def get_master_address(self): | ||
return await self.proxy.get_master_address() | ||
|
||
def rotate_slaves(self) -> AsyncIterator: | ||
"""Round-robin slave balancer""" | ||
return self.proxy.rotate_slaves() | ||
|
||
|
||
class Sentinel(AsyncSentinelCommands): | ||
|
@@ -318,7 +424,10 @@ def master_for( | |
self, | ||
service_name: str, | ||
redis_class: Type[Redis] = Redis, | ||
connection_pool_class: Type[SentinelConnectionPool] = SentinelConnectionPool, | ||
connection_pool_class: Union[ | ||
Type[SentinelConnectionPool], | ||
Type[SentinelBlockingConnectionPool], | ||
] = SentinelConnectionPool, | ||
**kwargs, | ||
): | ||
""" | ||
|
@@ -355,7 +464,10 @@ def slave_for( | |
self, | ||
service_name: str, | ||
redis_class: Type[Redis] = Redis, | ||
connection_pool_class: Type[SentinelConnectionPool] = SentinelConnectionPool, | ||
connection_pool_class: Union[ | ||
Type[SentinelConnectionPool], | ||
Type[SentinelBlockingConnectionPool], | ||
] = SentinelConnectionPool, | ||
**kwargs, | ||
): | ||
""" | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
What is the purpose of this proxy class?
get_master_address
androtate_slaves
methods implementations are exactly the same as inSentinelConnectionPool
class. The only difference that I see is the missing calls to parent object in constructor and reset.There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I did it this way because it is already done in the synchronous version: https://github.com/redis/redis-py/blob/master/redis/sentinel.py#L89
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yeah, I get it. It's a bad design approach and I don't want to spread it even more. It should be implemented correctly and I will create an issue for the future to do the same with sync version.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Okay. I'll think about a better solution
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The best solution will be refactor async ConnectionPool the way we don't need to use workarounds to break inheritance limitations
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Hello. What if we move BlockingConnectionPool logic to ConnectionPool?
We can do something like this: