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 не получит строку перевода, в то время, когда цикл свободен для выполнения других команд, если они существуют.