我有一个像这样的简单服务器:
#!/usr/bin/env python2
import socket
server = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
server.bind(('localhost', 50000))
server.listen(5)
# Set TCP Keepalive
server.setsockopt(socket.SOL_SOCKET, socket.SO_KEEPALIVE, 1)
server.setsockopt(socket.IPPROTO_TCP, socket.TCP_KEEPIDLE, 1)
server.setsockopt(socket.IPPROTO_TCP, socket.TCP_KEEPINTVL, 3)
server.setsockopt(socket.IPPROTO_TCP, socket.TCP_KEEPCNT, 5)
connection, client_address = server.accept()
while True:
try:
data = connection.recv(1024)
except socket.error:
print "%s dropped out" % (connection.getpeername(),)
break
if data:
print "%s: %s" % (connection.getpeername(), data)
else:
print "%s disconnected" % (connection.getpeername(),)
break
我使用 TCP Keepalive 来确定客户端是否消失(而不是正确断开连接)。
我看到的问题是,当一个客户端确实消失了,并且我捕获了 socket.error 异常时,我的打印语句会尝试通过使用 connection.getpeername() 来确定哪个客户端断开连接,这不再有效。它抛出一个
socket.error: [Errno 107] Transport endpoint is not connected
例外。
在这个简单的例子中,任何时候只有一个客户端可以连接,但是如果有更复杂的代码同时处理多个客户端,知道哪个客户端退出可能会很有用。
一旦套接字断开连接,是否无法知道它之前连接到哪个端点?为什么即使在客户端断开连接后,connection对象也不保留客户端的 IP/端口信息connection.getpeername()?这是因为该connection对象可能是可重用的吗?
我怎样才能让我的代码真正工作,并打印出哪个客户端退出了?我是否需要将 的输出存储connection.getpeername()在一个变量中,以便我以后可以调用它?这意味着如果我有多个客户端,我需要将connection.getpeername()每个客户端的值存储在列表中吗?
我知道使用socket是相当低级的,而且我很确定实际使用SocketServer.TCPServer会更容易 - 我只是想了解所涉及的低级网络。
阿波罗的战车
相关分类