Browse Source

Made aio_pika client manager more robust and efficient (Fixes #1142)

pull/1163/head
Miguel Grinberg 2 years ago
parent
commit
cd7f781c02
No known key found for this signature in database GPG Key ID: 36848B262DF5F06C
  1. 98
      src/socketio/asyncio_aiopika_manager.py

98
src/socketio/asyncio_aiopika_manager.py

@ -44,9 +44,10 @@ class AsyncAioPikaManager(AsyncPubSubManager): # pragma: no cover
'(Run "pip install aio_pika" in your ' '(Run "pip install aio_pika" in your '
'virtualenv).') 'virtualenv).')
self.url = url self.url = url
self.listener_connection = None self._lock = asyncio.Lock()
self.listener_channel = None self.publisher_connection = None
self.listener_queue = None self.publisher_channel = None
self.publisher_exchange = None
super().__init__(channel=channel, write_only=write_only, logger=logger) super().__init__(channel=channel, write_only=write_only, logger=logger)
async def _connection(self): async def _connection(self):
@ -66,41 +67,60 @@ class AsyncAioPikaManager(AsyncPubSubManager): # pragma: no cover
return queue return queue
async def _publish(self, data): async def _publish(self, data):
connection = await self._connection() if self.publisher_connection is None:
channel = await self._channel(connection) async with self._lock:
exchange = await self._exchange(channel) if self.publisher_connection is None:
await exchange.publish( self.publisher_connection = await self._connection()
aio_pika.Message(body=pickle.dumps(data), self.publisher_channel = await self._channel(
delivery_mode=aio_pika.DeliveryMode.PERSISTENT), self.publisher_connection
routing_key='*'
)
async def _listen(self):
retry_sleep = 1
while True:
try:
if self.listener_connection is None:
self.listener_connection = await self._connection()
self.listener_channel = await self._channel(
self.listener_connection
) )
await self.listener_channel.set_qos(prefetch_count=1) self.publisher_exchange = await self._exchange(
exchange = await self._exchange(self.listener_channel) self.publisher_channel
self.listener_queue = await self._queue(
self.listener_channel, exchange
) )
retry_sleep = 1 retry = True
while True:
async with self.listener_queue.iterator() as queue_iter: try:
async for message in queue_iter: await self.publisher_exchange.publish(
async with message.process(): aio_pika.Message(
yield pickle.loads(message.body) body=pickle.dumps(data),
except Exception: delivery_mode=aio_pika.DeliveryMode.PERSISTENT
self._get_logger().error('Cannot receive from rabbitmq... ' ), routing_key='*',
'retrying in ' )
'{} secs'.format(retry_sleep)) return
self.listener_connection = None except aio_pika.AMQPException:
await asyncio.sleep(retry_sleep) if retry:
retry_sleep *= 2 self._get_logger().error('Cannot publish to rabbitmq... '
if retry_sleep > 60: 'retrying')
retry_sleep = 60 retry = False
else:
self._get_logger().error(
'Cannot publish to rabbitmq... giving up')
break
except aio_pika.exceptions.ChannelInvalidStateError:
# aio_pika raises this exception when the task is cancelled
raise asyncio.CancelledError()
async def _listen(self):
async with (await self._connection()) as connection:
channel = await self._channel(connection)
await channel.set_qos(prefetch_count=1)
exchange = await self._exchange(channel)
queue = await self._queue(channel, exchange)
retry_sleep = 1
while True:
try:
async with queue.iterator() as queue_iter:
async for message in queue_iter:
async with message.process():
yield pickle.loads(message.body)
retry_sleep = 1
except aio_pika.AMQPException:
self._get_logger().error(
'Cannot receive from rabbitmq... '
'retrying in {} secs'.format(retry_sleep))
await asyncio.sleep(retry_sleep)
retry_sleep = min(retry_sleep * 2, 60)
except aio_pika.exceptions.ChannelInvalidStateError:
# aio_pika raises this exception when the task is cancelled
raise asyncio.CancelledError()

Loading…
Cancel
Save