对 python asyncio gRPC 客户端的多线程支持

问题描述 投票:0回答:1

我有一个在多线程环境中使用的 asyncio gRPC 客户端。当多个线程同时通过客户端连接到服务时,我看到以下错误流:

2021-01-27 09:33:56,937 错误 [asyncio] [thread_0] 回调 PollerCompletionQueue._handle_events()() 中出现异常
处理: )()>
回溯(最近一次调用最后一次):
  文件“/usr/local/lib/python3.8/asyncio/events.py”,第 81 行,在 _run 中
    self._context.run(self._callback, *self._args)
  文件“src/python/grpcio/grpc/_cython/_cygrpc/aio/completion_queue.pyx.pxi”,第 147 行,位于 grpc._cython.cygrpc.PollerCompletionQueue._handle_events 中
BlockingIOError:[Errno 11] 资源暂时不可用
2021-01-27 09:33:56,937 错误 [asyncio] [thread_1] 回调 PollerCompletionQueue._handle_events()() 中出现异常
处理: )()>
回溯(最近一次调用最后一次):
  文件“/usr/local/lib/python3.8/asyncio/events.py”,第 81 行,在 _run 中
    self._context.run(self._callback, *self._args)
  文件“src/python/grpcio/grpc/_cython/_cygrpc/aio/completion_queue.pyx.pxi”,第 147 行,位于 grpc._cython.cygrpc.PollerCompletionQueue._handle_events 中
BlockingIOError:[Errno 11] 资源暂时不可用

请求似乎已成功完成,但是,消息淹没了我的日志,让我感到紧张!

在我的测试中,每个线程创建自己的通道并提交自己的异步请求。无论服务负载如何,都会发生错误。如果客户端在不同的进程中运行,则不会发生错误。

我的设置:

  • Python版本:
    3.8.6
  • grpcio
    版本:
    1.35.0

任何见解表示赞赏!

python python-asyncio grpc python-multithreading grpc-python
1个回答
5
投票

gRPC AsyncIO 使用 UDS 在 C 扩展和 Python 之间进行通信。从您的日志来看,fd 访问存在竞争条件。 AsyncIO API 支持多线程,但这看起来像是一个新问题(它有助于在 https://github.com/grpc/grpc/issues 上创建问题)。

修复竞争条件可能很棘手,因为 AsyncIO 使用非线程安全的 AsyncIO 锁。如果我们使用线程安全锁保护 fd,它可能会阻塞 AsyncIO 循环。请随意提出或贡献解决方案。

如果让所有客户端在一个线程上运行,AsyncIO 的性能最佳。事件循环将很好地处理协程的执行,而无需线程跳跃。如果目标是使机器上的所有计算能力饱和,正如您提到的,最好使用多重处理。

链接到基本 gRPC AsyncIO 示例:https://github.com/grpc/grpc/blob/master/examples/python/helloworld/async_greeter_client.py

© www.soinside.com 2019 - 2024. All rights reserved.