Python telnetlib:令人惊讶的问题

ptz*_*ptz 3 python telnet telnetlib

我正在使用Python模块telnetlib来创建一个telnet会话(使用国际象棋服务器),我遇到了一个问题,我真的无法包围我的大脑.以下代码完美运行:

>>> f = login("my_server") #code for login(host) below.
>>> f.read_very_eager()
Run Code Online (Sandbox Code Playgroud)

这会吐出服务器通常在登录时打印的所有内容.但是,当我把它放在一个函数中,然后调用它:

>>> def foo():
...   f = login("my_server")
...   return f.read_very_eager()
...
>>> foo()
Run Code Online (Sandbox Code Playgroud)

我什么都没得到(空字符串).我可以检查登录是否正确执行,但由于某种原因我看不到文本.它吞噬了哪里?

非常感谢.

为了完整性,这里是登录(主机):

def login(host, handle="guest", password=""):
try:
    f = telnetlib.Telnet(host) #connect to host
except:
    raise Error("Could not connect to host")
f.read_until("login: ")
try:
    f.write(handle + "\n\r")
except:
    raise Error("Could not write username to host")
if handle == "guest":
    f.read_until(":\n\r")
else:
    f.read_until("password: ")
try:
    f.write(password + "\n\r")
except:
    raise Error("Could not write password to host")
return f
Run Code Online (Sandbox Code Playgroud)

Mew*_*Mew 5

当你手动尝试而不是在函数中时,这是因为当你手动尝试时,服务器有足够的时间对登录做出反应并发回数据.当它在一个函数中时,您将密码发送到服务器,并且永远不会等待服务器回复.

如果您更喜欢(可能更正确)技术答案:

在文件telnetlib.py(我的Windows计算机上的c:\ python26\Lib\telnetlib.py)中,函数read_very_eager(self)调用self.sock_avail()Now,函数sock_avail(self)执行以下操作:

def sock_avail(self):
    """Test whether data is available on the socket."""
    return select.select([self], [], [], 0) == ([self], [], [])
Run Code Online (Sandbox Code Playgroud)

这样做非常简单:如果有任何东西 - 从我们的套接字读取(服务器已经回答),它将返回True,否则它将返回False.

那么,read_very_eager(self)它是什么:检查是否有任何可供阅读的内容.如果有,则从套接字读取,否则只返回一个空字符串.

如果你查看你的代码,read_some(self)你会看到它没有检查是否有任何数据可供阅读.它会尝试阅读,直到有可用的东西,这意味着如果服务器在回答你之前需要100毫秒,它会在返回答案之前等待100毫秒.