2020-01-30 09:07:26 +08:00
|
|
|
import asyncio
|
|
|
|
import pickle
|
|
|
|
|
2022-11-08 02:06:49 +08:00
|
|
|
try: # pragma: no cover
|
|
|
|
from redis import asyncio as aioredis
|
|
|
|
from redis.exceptions import RedisError
|
|
|
|
except ImportError: # pragma: no cover
|
|
|
|
try:
|
|
|
|
import aioredis
|
|
|
|
from aioredis.exceptions import RedisError
|
|
|
|
except ImportError:
|
|
|
|
aioredis = None
|
|
|
|
RedisError = None
|
2020-01-30 09:07:26 +08:00
|
|
|
|
|
|
|
from .asyncio_pubsub_manager import AsyncPubSubManager
|
|
|
|
|
|
|
|
|
|
|
|
class AsyncRedisManager(AsyncPubSubManager): # pragma: no cover
|
|
|
|
"""Redis based client manager for asyncio servers.
|
|
|
|
|
|
|
|
This class implements a Redis backend for event sharing across multiple
|
2022-01-24 12:07:52 +08:00
|
|
|
processes.
|
2020-01-30 09:07:26 +08:00
|
|
|
|
2022-01-24 12:07:52 +08:00
|
|
|
To use a Redis backend, initialize the :class:`AsyncServer` instance as
|
2020-01-30 09:07:26 +08:00
|
|
|
follows::
|
|
|
|
|
2022-01-24 12:07:52 +08:00
|
|
|
url = 'redis://hostname:port/0'
|
|
|
|
server = socketio.AsyncServer(
|
|
|
|
client_manager=socketio.AsyncRedisManager(url))
|
2020-01-30 09:07:26 +08:00
|
|
|
|
|
|
|
:param url: The connection URL for the Redis server. For a default Redis
|
|
|
|
store running on the same host, use ``redis://``. To use an
|
|
|
|
SSL connection, use ``rediss://``.
|
|
|
|
:param channel: The channel name on which the server sends and receives
|
|
|
|
notifications. Must be the same in all the servers.
|
2021-05-08 22:25:29 +08:00
|
|
|
:param write_only: If set to ``True``, only initialize to emit events. The
|
2020-01-30 09:07:26 +08:00
|
|
|
default of ``False`` initializes the class for emitting
|
|
|
|
and receiving.
|
2022-01-24 12:07:52 +08:00
|
|
|
:param redis_options: additional keyword arguments to be passed to
|
|
|
|
``aioredis.from_url()``.
|
2020-01-30 09:07:26 +08:00
|
|
|
"""
|
|
|
|
name = 'aioredis'
|
|
|
|
|
|
|
|
def __init__(self, url='redis://localhost:6379/0', channel='socketio',
|
2022-01-24 12:07:52 +08:00
|
|
|
write_only=False, logger=None, redis_options=None):
|
2020-01-30 09:07:26 +08:00
|
|
|
if aioredis is None:
|
|
|
|
raise RuntimeError('Redis package is not installed '
|
2022-11-08 02:06:49 +08:00
|
|
|
'(Run "pip install redis" in your virtualenv).')
|
2022-01-24 12:07:52 +08:00
|
|
|
if not hasattr(aioredis.Redis, 'from_url'):
|
|
|
|
raise RuntimeError('Version 2 of aioredis package is required.')
|
|
|
|
self.redis_url = url
|
|
|
|
self.redis_options = redis_options or {}
|
|
|
|
self._redis_connect()
|
2020-01-30 09:07:26 +08:00
|
|
|
super().__init__(channel=channel, write_only=write_only, logger=logger)
|
|
|
|
|
2022-01-24 12:07:52 +08:00
|
|
|
def _redis_connect(self):
|
|
|
|
self.redis = aioredis.Redis.from_url(self.redis_url,
|
|
|
|
**self.redis_options)
|
|
|
|
self.pubsub = self.redis.pubsub(ignore_subscribe_messages=True)
|
|
|
|
|
2020-01-30 09:07:26 +08:00
|
|
|
async def _publish(self, data):
|
|
|
|
retry = True
|
|
|
|
while True:
|
|
|
|
try:
|
2022-01-24 12:07:52 +08:00
|
|
|
if not retry:
|
|
|
|
self._redis_connect()
|
|
|
|
return await self.redis.publish(
|
|
|
|
self.channel, pickle.dumps(data))
|
2022-11-08 02:06:49 +08:00
|
|
|
except RedisError:
|
2020-01-30 09:07:26 +08:00
|
|
|
if retry:
|
|
|
|
self._get_logger().error('Cannot publish to redis... '
|
|
|
|
'retrying')
|
|
|
|
retry = False
|
|
|
|
else:
|
|
|
|
self._get_logger().error('Cannot publish to redis... '
|
|
|
|
'giving up')
|
|
|
|
break
|
|
|
|
|
2022-01-24 12:07:52 +08:00
|
|
|
async def _redis_listen_with_retries(self):
|
2020-01-30 09:07:26 +08:00
|
|
|
retry_sleep = 1
|
2022-01-24 12:07:52 +08:00
|
|
|
connect = False
|
2020-01-30 09:07:26 +08:00
|
|
|
while True:
|
|
|
|
try:
|
2022-01-24 12:07:52 +08:00
|
|
|
if connect:
|
|
|
|
self._redis_connect()
|
|
|
|
await self.pubsub.subscribe(self.channel)
|
|
|
|
retry_sleep = 1
|
|
|
|
async for message in self.pubsub.listen():
|
|
|
|
yield message
|
2022-11-08 02:06:49 +08:00
|
|
|
except RedisError:
|
2020-01-30 09:07:26 +08:00
|
|
|
self._get_logger().error('Cannot receive from redis... '
|
|
|
|
'retrying in '
|
|
|
|
'{} secs'.format(retry_sleep))
|
2022-01-24 12:07:52 +08:00
|
|
|
connect = True
|
2020-01-30 09:07:26 +08:00
|
|
|
await asyncio.sleep(retry_sleep)
|
|
|
|
retry_sleep *= 2
|
|
|
|
if retry_sleep > 60:
|
|
|
|
retry_sleep = 60
|
2022-01-24 12:07:52 +08:00
|
|
|
|
|
|
|
async def _listen(self):
|
|
|
|
channel = self.channel.encode('utf-8')
|
|
|
|
await self.pubsub.subscribe(self.channel)
|
|
|
|
async for message in self._redis_listen_with_retries():
|
|
|
|
if message['channel'] == channel and \
|
|
|
|
message['type'] == 'message' and 'data' in message:
|
|
|
|
yield message['data']
|
|
|
|
await self.pubsub.unsubscribe(self.channel)
|