Aysncio не может читать stdin в Windows
Я пытаюсь читать stdin асинхронно в 64-битной версии Windows 7 и Python 3.4.3
Я попробовал это, опираясь на SO ответ:
import asyncio
import sys
def reader():
print('Received:', sys.stdin.readline())
loop = asyncio.get_event_loop()
task = loop.add_reader(sys.stdin.fileno(), reader)
loop.run_forever()
loop.close()
Однако он вызывает OSError: [WInError 100381] An operation was attempted on something that is not a socket
.
Может ли файл-подобный объект, например stdin
, быть завернут в класс, чтобы дать ему API сокета? Я задал этот вопрос отдельно, но если решение прост, ответьте здесь.
Предполагая, что я не могу обернуть объект, подобный файлу, чтобы сделать его сокетом, я попытался использовать потоки, вдохновленные this gist:
import asyncio
import sys
@asyncio.coroutine
def stdio(loop):
reader = asyncio.StreamReader(loop=loop)
reader_protocol = asyncio.StreamReaderProtocol(reader)
yield from loop.connect_read_pipe(lambda: reader_protocol, sys.stdin)
@asyncio.coroutine
def async_input(loop):
reader = yield from stdio(loop)
line = yield from reader.readline()
return line.decode().replace('\r', '').replace('\n', '')
@asyncio.coroutine
def main(loop):
name = yield from async_input(loop)
print('Hello ', name)
loop = asyncio.get_event_loop()
loop.run_until_complete(main(loop))
loop.close()
И это вызывает NotImplementedError
в asyncio.base_events._make_read_pipe_transport
Пожалуйста, сообщите, как читать stdin
с помощью asyncio
в Windows...
Ответы
Ответ 1
Исключено исключение NotImplementedError
, потому что connect pipe coroutines не поддерживается SelectorEventLoop
, который представляет собой набор циклов событий по умолчанию на asyncio
. Для поддержки труб в Windows вам необходимо использовать ProactorEventLoop
. Тем не менее, он все равно не будет работать, потому что, по-видимому, функции connect_read_pipe
и connect_write_pipe
не поддерживают stdin
/stdout
/stderr
или файлы в Windows как Python 3.5.1.
Один из способов чтения из stdin
с асинхронным поведением - использовать поток с помощью метода loop run_in_executor
. Вот простой пример для справки:
import asyncio
import sys
async def aio_readline(loop):
while True:
line = await loop.run_in_executor(None, sys.stdin.readline)
print('Got line:', line, end='')
loop = asyncio.get_event_loop()
loop.run_until_complete(aio_readline(loop))
loop.close()
В примере функция sys.stdin.readline()
вызывается в другом потоке с помощью метода loop.run_in_executor
. Поток остается заблокированным до тех пор, пока stdin
не получит строку перевода, в то время, когда цикл свободен для выполнения других команд, если они существуют.