tcp远端命令执行--解决粘包

2019-07-15  本文已影响0人  温柔倾怀

客户端服务端通信简图

两种粘包现象

解决粘包现象

只有tcp通信存在粘包现象
udp会直接产生OSError异常

客户端

from socket import *
import struct
from functools import partial

ip_port = ('127.0.0.1',8080)
back_log =5
buffer_size = 1024

tcp_client = socket(AF_INET,SOCK_STREAM)
tcp_client.connect(ip_port)

while True:
    cmd = input(">>:").strip()
    if not cmd:continue
    if cmd == 'quit':break
    tcp_client.send(cmd.encode('utf8'))

    #解决粘包
    length_data = tcp_client.recv(4) #接受数据长度
    length = struct.unpack('i',length_data)[0]
    #开始接受数据
    # recv_size = 0
    # recv_msg = b''
    # while recv_size<length:
    #     recv_msg += tcp_client.recv(buffer_size)
    #     recv_size = len(recv_msg)
    #牛逼的代码--这里实现有些问题--学习这种牛逼的写法
    recv_msg = ''.join(iter(partial(tcp_client.recv,buffer_size),b''))

    print(recv_msg.decode('gbk'))
tcp_client.close()

服务端


from socket import *
import subprocess
import struct

ip_port = ('127.0.0.1',8080)
back_log =5
buffer_size = 1024

tcp_server = socket(AF_INET,SOCK_STREAM)
tcp_server.bind(ip_port)
tcp_server.listen(back_log)

while True:
    conn,addr = tcp_server.accept()
    print("新的客户端",addr)
    while True:
        # 接收到的是客户端发来的命令,服务端执行,并返回结果
        try:
            cmd = conn.recv(buffer_size).decode('utf8')
            if not cmd:break
            print("客户端命令:",cmd)
            #执行命令
            res = subprocess.Popen(cmd,shell=True,
                                   stdout=subprocess.PIPE,
                                   stdin=subprocess.PIPE,
                                   stderr=subprocess.PIPE)
            err = res.stderr.read()
            if err:
                cmd_res = err
            else:
                cmd_res = res.stdout.read()
            # 命令正常执行没有返回值得情况下,比如cd ..
            if not cmd_res:
                cmd_res = 'exec success'.encode('gbk')
            #解决粘包
            length = len(cmd_res)
            data_length = struct.pack('i',length)
            #下面两个send会黏到一起;在客户端解决:第一次接受数据长度;第二次开始接受数据
            #定制报文头
            conn.send(data_length)
            #真实发送数据
            conn.send(cmd_res)
        except ConnectionResetError as e:
            print(e)
            break
    conn.close()
tcp_server.close()

上一篇 下一篇

猜你喜欢

热点阅读