简单的python asyncio服务器不读取数据

时间:2018-02-18 21:13:38

标签: python-asyncio

我在windows10上使用python3.6和asyncio来制作一个简单的服务器。我有问题让服务器以我想要的方式工作。 但是使用套接字接口实现的同一服务器可以正常工作。 这是工作服务器

import socket

with socket.socket(socket.AF_INET, socket.SOCK_STREAM) as s:
    s.bind(('', 3201))
    s.listen(1)
    conn, addr = s.accept()
    with conn:
        print('Client connection: {}:{}'.format(*addr))
        while True:
            data = conn.recv(1024)
            if data:
                print('recv {} bytes'.format(len(data)))
            else:
                print('no data')
                break
print('connection closed')

我用nc命令测试它。 e.g。

nc 10.68.100.32 3201
abc
ab
a
^C

服务器打印:

Client connection: 10.68.101.87:55243
recv 4 bytes
recv 3 bytes
recv 2 bytes
no data
connection closed    

但是我不能用asyncio做同样的服务器。 这是一个协议版本:

import asyncio

class ListenServer(asyncio.Protocol):

    def connection_made(self, transport):
        address = transport.get_extra_info('peername')
        print('connection made by client:{}:{}'.format(*address))

    def data_deceived(self, data):
        print('received {}'.format(len(data)), flush=True)

    def eof_received(self):
        print('print received EOF')

    def connection_lost(self, error):
        if error:
            print('ERROR: {}'.format(error))
        else:
            print('connection closed')

event_loop = asyncio.get_event_loop()
factory = event_loop.create_server(ListenServer, '', 3201)
server = event_loop.run_until_complete(factory)

event_loop.run_forever()
print('closing server')
server.close()

对于与上述服务器写入相同的nc测试序列:

connection made by client:10.68.101.87:55247
print received EOF
connection closed

我也失败了服务器的asyncio coroutine API版本:

import asyncio

async def just_read(reader, writer):
    cli_host, cli_port = writer.get_extra_info('peername')
    print('Client connected {}:{}'.format(cli_host, cli_port))
    while True:
        data = await reader.read()
        if data:
            print('received {} bytes'.format(len(data)), flush=True)
        else:
            print('closing')
            writer.close()
            return

event_loop = asyncio.get_event_loop()
coro = asyncio.start_server(just_read, '', 3201, loop=event_loop)
server = event_loop.run_until_complete(coro)

event_loop.run_forever()
server.close()
print('Connection closed')

在这种情况下的输出是一个小小的差异:

Client connected 10.68.101.87:55253
received 9 bytes
closing

在这种情况下,当我用^ C关闭nc时,我得到所有数据, 并且服务器继续,以便我可以再次连接。

我希望服务器读取并打印出客户端写的内容, 只要客户端连接,它应连接一次并保持连接。 如何使用asyncio实现这一点,我应该使用协议或协程版本,这有关系吗?

这只是我希望服务器执行的几项任务中的第一项,也可能是最简单的。 此处未示出的其他任务是开始定期发送虚拟消息 并发送不同消息的响应。

1 个答案:

答案 0 :(得分:1)

协议代码包含拼写错误,它将方法命名为data_deceived而不是data_received。方法名称更正后,输出将与阻止示例中的输出相同。

在协程版本中,问题是reader.read()指示asyncio收集数据直到文件结束,并在准备好后提供。将其更改为reader.read(1024),与阻止版本中的sock.recv(1024)类似,修复代码。