我有一个使用 asyncio 运行的进程,它应该永远运行。

我可以使用 ProcessIterator 与该进程交互,它可以(此处省略)将数据发送到 stdin 并从 stdout 获取。

我可以使用 async for fd, data in ProcessIterator(...): 访问数据。

现在的问题是这个异步迭代器的执行必须是有时间限制的。如果时间用完,则调用 timeout() 函数,
但异常并非源自 __anext__ 函数来通知超时。

如何在异步迭代器中引发此异常?
我发现无法调用 awaitable.throw(something) 或类似的方法。

class ProcessIterator:
    def __init__(self, process, loop, run_timeout):
        self.process = process
        self.loop = loop

        self.run_timeout = run_timeout

        # set the global timer
        self.overall_timer = self.loop.call_later(
            self.run_timeout, self.timeout)

    def timeout(self):
        # XXX: how do i pass this exception into the iterator?
        raise ProcTimeoutError(
            self.process.args,
            self.run_timeout,
            was_global,
        )

    async def __aiter__(self):
        return self

    async def __anext__(self):
        if self.process.exited:
            raise StopAsyncIteration()

        else:
            # fetch output from the process asyncio.Queue()
            entry = await self.process.output_queue.get()
            if entry == StopIteration:
                raise StopAsyncIteration()

            return entry

异步迭代器的用法现在大致是:
async def test_coro(loop):
    code = 'print("rofl"); time.sleep(5); print("lol")'

    proc = Process([sys.executable, '-u', '-c', code])

    await proc.create()

    try:
        async for fd, line in ProcessIterator(proc, loop, run_timeout=1):
            print("%d: %s" % (fd, line))

    except ProcessTimeoutError as exc:
        # XXX This is the exception I'd like to get here! How can i throw it?
        print("timeout: %s" % exc)

    await proc.wait()

tl; dr:如何抛出一个定时异常,使其源自异步迭代器?

最佳答案

编辑:添加了解决方案 2

解决方案 1:
timeout() 回调可以将 ProcTimeoutError 异常存储在实例变量中吗?然后 __anext__() 可以检查实例变量并在设置时引发异常。

class ProcessIterator:
    def __init__(self, process, loop, run_timeout):
        self.process = process
        self.loop = loop
        self.error = None

        self.run_timeout = run_timeout

        # set the global timer
        self.overall_timer = self.loop.call_later(
            self.run_timeout, self.timeout)

    def timeout(self):
        # XXX: set instance variable
        self.error = ProcTimeoutError(
                         self.process.args,
                         self.run_timeout,
                         was_global
                     )

    async def __aiter__(self):
        return self

    async def __anext__(self):
        # XXX: if error is set, then raise the exception
        if self.error:
            raise self.error

        elif self.process.exited:
            raise StopAsyncIteration()

        else:
            # fetch output from the process asyncio.Queue()
            entry = await self.process.output_queue.get()
            if entry == StopIteration:
                raise StopAsyncIteration()

            return entry

解决方案 2:

将异常放在 process.output_queue 上。
....
def timeout(self):
    # XXX: set instance variable
    self.process.ouput_queue.put(ProcTimeoutError(
                                     self.process.args,
                                     self.run_timeout,
                                     was_global
                                 ))

....

# fetch output from the process asyncio.Queue()
entry = await self.process.output_queue.get()
if entry == StopIteration:
    raise StopAsyncIteration()

elif entry = ProcTimeoutError:
    raise entry
....

如果队列中可能有条目,请使用优先级队列。为 ProcTimeoutError 分配比其他条目更高的优先级,例如 (0, ProcTimeoutError) vs (1, other_entry)。

关于python - 通过超时取消异步迭代器,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/35880234/

10-11 19:33