模拟ssh远程执行命令

122-模拟ssh远程执行命令-ssh.png

服务端

from socket import *
import subprocess
import struct

server = socket(AF_INET, SOCK_STREAM)
server.bind(('127.0.0.1', 8080))
server.listen(5)

while True:
    conn, client_addr = server.accept()
    print('新的客户端', client_addr)

    while True:
        try:
            cmd = conn.recv(1024)  # cmd=b'dir'
            if len(cmd) == 0: break

            # 运行系统命令
            obj = subprocess.Popen(cmd.decode('utf-8'),
                                   shell=True,
                                   stderr=subprocess.PIPE,
                                   stdout=subprocess.PIPE)

            stdout = obj.stdout.read()
            stderr = obj.stderr.read()

            # 1、先制作报头(固定长度)
            total_size = len(stdout) + len(stderr)

            header = struct.pack('i', total_size)

            # 2、先发送固定长度的报头
            conn.send(header)

            # 3、再发送真实的数据
            conn.send(stdout)
            conn.send(stderr)
        except ConnectionResetError:
            break

    conn.close()

客户端1

from socket import *
import struct

client = socket(AF_INET, SOCK_STREAM)
client.connect(('127.0.0.1', 8080))

while True:
    cmd = input('>>: ').strip()
    if len(cmd) == 0: continue
    client.send(cmd.encode('utf-8'))

    #1、先收固定长度的报头
    header = client.recv(4)

    #2、从报头中解析出对数据的描述信息
    total_size = struct.unpack('i', header)[0]

    #3、再收真实的数据
    recv_size = 0
    res = b''
    while recv_size < total_size:
        data = client.recv(1024)
        res += data
        recv_size += len(data)
    # subprocess.Popen()的结果的编码是以当前所在的系统为准的,如果是windows,那么res.stdout.read()读出的就是GBK编码的,在接收端需要用GBK解码
    print(res.decode('gbk'))
  • 输入dir命令,由于服务端发送字节少于1024字节,客户端可以接受

  • 输入tasklist命令,由于服务端发送字节多于1024字节,客户端只接受部分数据,并且当你再次输入dir命令的时候,客户端会接收dir命令的结果,但是会打印上一次的剩余未发送完的数据,这就是粘包问题

猜你喜欢

转载自www.cnblogs.com/nickchen121/p/11030880.html