在此异步设置中,我在哪里捕获KeyboardInterrupt异常

时间:2019-02-04 23:29:58

标签: python asynchronous python-asyncio ccxt

我正在开发一个使用ccxt异步库的项目,该库要求通过显式调用该类的.close()协程来释放某个类使用的所有资源。我想用ctrl+c退出程序,并在异常中等待关闭协程。但是,从未等待过它。

应用程序由模块harvestersstrategiestradersbrokermain(加上配置等)组成。经纪人启动为交换指定的策略并执行它们。该策略启动关联的收割机,该收割机收集必要的数据。它还分析数据并在有获利机会时产生交易者。主模块为每个交易所创建一个代理并运行它。我曾尝试在每个级别上捕获异常,但从未等待过关闭例程。我更喜欢在主模块中捕获它,以关闭所有交换实例。

收割者

async def harvest(self):
    if not self.routes:
        self.routes = await self.get_routes()
    for route in self.routes:
        self.logger.info("Harvesting route {}".format(route))
        await asyncio.sleep(self.exchange.rateLimit / 1000)
        yield await self.harvest_route(route)

策略

async def execute(self):
    async for route_dct in self.harvester.harvest():
        self.logger.debug("Route dictionary: {}".format(route_dct))
        await self.try_route(route_dct)

经纪人

async def run(self):
    for strategy in self.strategies:
        self.strategies[strategy] = getattr(
            strategies, strategy)(self.share, self.exchange, self.currency)
    while True:
        try:
            await self.execute_strategies()
        except KeyboardInterrupt:
            await safe_exit(self.exchange)

主要

async def main():
    await load_exchanges()
    await load_markets()
    brokers = [Broker(
        share,
        exchanges[id]["api"],
        currency,
        exchanges[id]["strategies"]
        ) for id in exchanges]
    futures = [broker.run() for broker in brokers]
    for future in asyncio.as_completed(futures):
        executed = await future
        return executed


if __name__ == "__main__":
    status = asyncio.run(main())
    sys.exit(status)

我原本希望等待close()协程,但是我仍然从库中得到一个错误,必须明确调用它。在哪里捕获异常,以便所有交换实例正确关闭?

1 个答案:

答案 0 :(得分:1)

代码中的某个地方应该是事件循环开始的入口点。

通常它是以下功能之一:

loop.run_until_complete(main())

loop.run_forever()

asyncio.run(main())

ctrl+C发生时,KeyboardInterrupt可以在此行被捕获。当它执行一些定型协程时,您可以再次运行事件循环。

这个小例子说明了这个主意:

import asyncio 


async def main():
    print('Started, press ctrl+C')
    await asyncio.sleep(10)


async def close():
    print('Finalazing...')
    await asyncio.sleep(1)


if __name__ == '__main__':
    loop = asyncio.get_event_loop()
    try:
        loop.run_until_complete(main())
    except KeyboardInterrupt:
        loop.run_until_complete(close())
    finally:
        print('Program finished')