Хорошая ли идея использовать create_future asyncio?

#python #rabbitmq #python-asyncio

Вопрос:

Документация гласит: Обычно нет необходимости создавать будущие объекты в коде прикладного уровня. Но так часто мы видим следующий код (простой клиент RPC rabbitmq):

 import asyncio
import uuid
from aio_pika import connect, IncomingMessage, Message


class FibonacciRpcClient:
    def __init__(self, loop):
        self.connection = None
        self.channel = None
        self.callback_queue = None
        self.futures = {}
        self.loop = loop

    async def connect(self):
        self.connection = await connect(
            "amqp://guest:guest@localhost/", loop=loop
        )
        self.channel = await self.connection.channel()
        self.callback_queue = await self.channel.declare_queue(exclusive=True)
        await self.callback_queue.consume(self.on_response)

        return self

    def on_response(self, message: IncomingMessage):
        future = self.futures.pop(message.correlation_id)
        future.set_result(message.body)

    async def call(self, n):
        correlation_id = str(uuid.uuid4())
        future = self.loop.create_future()

        self.futures[correlation_id] = future

        await self.channel.default_exchange.publish(
            Message(
                str(n).encode(),
                content_type="text/plain",
                correlation_id=correlation_id,
                reply_to=self.callback_queue.name,
            ),
            routing_key="rpc_queue",
        )

        return int(await future)


async def main(loop):
    fibonacci_rpc = await FibonacciRpcClient(loop).connect()
    print(" [x] Requesting fib(30)")
    response = await fibonacci_rpc.call(30)
    print(" [.] Got %r" % response)


if __name__ == "__main__":
    loop = asyncio.get_event_loop()
    loop.run_until_complete(main(loop))
 

Хорошая ли идея использовать loop.create_future() в приведенном выше случае? Является ли это лучшей практикой для использования этого?

Ответ №1:

Обычно нет необходимости звонить create_future . В документах рекомендуется исключить эту функцию из любых интерфейсов API, ориентированных на пользователей.

Но в примере, которым вы поделились, метод используется для правильной выборки результата синхронной операции всеми другими способами on_response .

Фьючерсы могут/должны использоваться при написании асинхронных оболочек поверх других библиотек, когда возникает необходимость в объединении синхронных интерфейсов с async/await .