Python套接字 - 将数据包发送到服务器并等待响应

时间:2014-11-05 22:17:03

标签: python python-sockets

我目前正在实施以下代码,以便将数据包发送到 Parallels PVA XML API 。我正在尝试将两个 XML数据包发送到服务器。第一个是指定用户凭据的登录数据包,第二个是包含有关API请求的信息的数据包。这些数据包应该由空字节\0分隔。通常,服务器发回多个数据包,第一个是澄清登录成功,第二个包含有关API请求的信息。

我遇到的问题是,似乎第二个数据包似乎没有被发送。我收到的唯一响应是第一个说明登录成功的数据包,但我没有收到包含有关API请求信息的数据包。我认为这可能是因为我发送了一个空字节,所以我尝试在base64中对所有内容进行编码,但最终得到了相同的结果。

所以在我看来,连接正在关闭,服务器没有足够的时间发送它的第二个数据包,或者由于空字节而完全忽略了数据包。 / p>

非常感谢任何帮助或评论。提前谢谢!

import socket
import base64

def client(string):
    HOST, PORT = '[IP_ADDRESS]', 4433
    sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
    sock.settimeout(10)
    sock.connect((HOST, PORT))
    sock.send(base64.b64encode(string))
    reply = sock.recv(131072)
    sock.close()

    return reply

packet = "<packet></packet>\0<packet></packet>"
print client(packet)

请注意,数据包中没有任何信息,因为它包含敏感信息,并且IP地址被替换为&#34; [IP_ADDRESS]&#34;故意

2 个答案:

答案 0 :(得分:4)

问题是,您尝试一次发送太多字节,\0将终止整个消息,因此剩余部分从未发送过。 我模仿了一个类似的客户端/服务器,这是响应:

服务器只是一个回声类型

...

客户端,来自您的代码稍作修改

import socket
from time import sleep
def client():
    # I change second packet to 2packet for visually distinguish the packets
    packet = "<packet></packet>\0<2packet></2packet>"
    HOST, PORT = '127.0.0.1', 4433
    sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
    sock.settimeout(10)
    sock.connect((HOST, PORT))
    while True:
        try:
            sock.send(packet)
            sleep(1)
            # this is the problem here
            reply = sock.recv(131072)
            if not reply:
                break
            print "recvd: ", reply
        except KeyboardInterrupt:
            print "bye"
            break
    sock.close()
    return

client()
recvd:  WelcomeOK...<packet></packet> # the null byte teminate the send and skip all remaining message
recvd:  OK...<packet></packet>
recvd:  OK...<packet></packet>
recvd:  OK...<packet></packet>
...

你会看到, 2packet 从未被发送,因为它被空字节终止,消息体超过 131072 字节,这比第二部分消息体长

因此,为了解决这个问题,你只需要在每个循环上发送一个字节,因此空字节只会自行终止,并且你的下一个字节可以发送到消息结束:

通过发送单字节

修订版本
... previous code
    while True:
        try:
            sock.send(packet)
            sleep(1)
            reply = sock.recv(1)
            if not reply:
                break
            print "recvd: ", reply
        except KeyboardInterrupt:
            print "bye"
            break
    sock.close()
    return

client()
recvd:  W
recvd:  e
recvd:  l
recvd:  c
recvd:  o
recvd:  m
recvd:  e
recvd:  O
recvd:  K
recvd:  .
recvd:  .
recvd:  .
recvd:  <
recvd:  p
recvd:  a
recvd:  c
recvd:  k
recvd:  e
recvd:  t
recvd:  >
recvd:  <
recvd:  /
recvd:  p
recvd:  a
recvd:  c
recvd:  k
recvd:  e
recvd:  t
recvd:  >
recvd:       # <== this null byte terminates single byte send
recvd:  <    # <== and next loop it tries to send the next byte, goal achieved
recvd:  2
recvd:  p
recvd:  a
recvd:  c
recvd:  k
recvd:  e
recvd:  t
recvd:  >
recvd:  <
...

答案 1 :(得分:0)

sock.recv最多可以达到131072个字节,但它只能读取1个字节。如果立即关闭套接字,则回复将丢失。您必须重复recv,直到您阅读了服务器发送的所有信息。所有数据包都是\0终止的吗?然后阅读,直到你得到预期的&#39; \ 0&#39;