• Paramiko和堡垒机实现


    一、Paramiko

    • paramiko模块,基于SSH用于连接远程服务器并执行相关操作。

    1、安装:pip install paramiko

    2、SSHClient:用于连接远程服务器并执行基本命令

    • 基于用户名密码连接
    import paramiko 
    # 创建SSH对象
    ssh = paramiko.SSHClient()
    # 允许连接不在know_hosts文件中的主机
    ssh.set_missing_host_key_policy(paramiko.AutoAddPolicy())
    # 连接服务器
    ssh.connect(hostname='192.168.1.200', port=2000, username='root', password='123') 
    # 执行命令
    stdin, stdout, stderr = ssh.exec_command('ls')
    # 获取命令结果
    result = stdout.read()   
    # 关闭连接
    ssh.close()
    #自己创建transport
    import paramiko
    transport = paramiko.Transport(('192.168.1.200', 2000))
    transport.connect(username='root', password='123')
    ssh = paramiko.SSHClient()
    ssh._transport = transport
    stdin, stdout, stderr = ssh.exec_command('df')
    print stdout.read()
    transport.close()

    3、SFTPClient:用于连接远程服务器并执行上传下载

    • 基于用户名密码上传下载:
    import paramiko 
    transport = paramiko.Transport(('192.168.1.200',2000))
    transport.connect(username='root',password='123')  
    sftp = paramiko.SFTPClient.from_transport(transport)
    # 将location.py 上传至服务器 /tmp/test.py
    sftp.put('/tmp/location.py', '/tmp/test.py')
    # 将remove_path 下载到本地 local_path
    sftp.get('remove_path', 'local_path') 
    transport.close()

    4、transport实例

    import paramiko
    class SSHConnection:
        def __init__(self,hostname='192.168.1.200',port=2000,username='root',pwd='123'):
            self.hostname = hostname
            self.port = port
            self.username = username
            self.pwd = pwd
            self._k = None
    
        def connect(self):
            '''连接服务器'''
            transport = paramiko.Transport((self.hostname,self.port))
            transport.connect(username=self.username,password=self.pwd)
            self.__transport = transport
    
        def close(self):
            '''关闭链接'''
            self.__transport.close()
    
        def cmd(self,command):
            '''执行命令'''
            ssh = paramiko.SSHClient()
            ssh._transport = self.__transport
            stdin,stdout,stderr = ssh.exec_command(command)
            ret = stdout.read()
            return ret
    
        def upload(self,local_path,target_path):
            '''上传文件'''
            sftp = paramiko.SFTPClient.from_transport(self.__transport)
            sftp.put(local_path,target_path)
    
        def run(self):
            '''主程序'''
            self.connect()
            ret =  self.cmd('ls')
            print(ret.decode('utf-8'))
            self.upload('ssh_client.py','/tmp/s1.py')
            self.close()
              
    if __name__ == "__main__":
        ssh = SSHConnection()
        ssh.run()
    transport实例

    二、堡垒机

    堡垒机执行流程:

    1. 管理员为用户在服务器上创建账号(将公钥放置服务器,或者使用用户名密码)
    2. 用户登陆堡垒机,输入堡垒机用户名密码,现实当前用户管理的服务器列表
    3. 用户选择服务器,并自动登陆
    4. 执行操作并同时将用户操作记录

    注:配置.brashrc实现ssh登陆后自动执行脚本,如:/usr/bin/python /home/wupeiqi/menu.py

    1、前戏

    import paramiko
    import sys
    import os
    import socket
    import select
    import getpass
     
    tran = paramiko.Transport(('192.168.1.200', 2000,))
    tran.start_client()
    tran.auth_password('root', '123')
     
    # 打开一个通道
    chan = tran.open_session()
    # 获取一个终端
    chan.get_pty()
    # 激活器
    chan.invoke_shell()
     
    #########
    # 利用sys.stdin,肆意妄为执行操作
    # 用户在终端输入内容,并将内容发送至远程服务器
    # 远程服务器执行命令,并将结果返回
    # 用户终端显示内容
    #########
     
    chan.close()
    tran.close()
    

    2、肆意妄为(一)

    import paramiko
    import sys
    import os
    import socket
    import select
    import getpass
    from paramiko.py3compat import u
     
    tran = paramiko.Transport(('192.168.1.200', 2000,))
    tran.start_client()
    tran.auth_password('root', '123')
     
    # 打开一个通道
    chan = tran.open_session()
    # 获取一个终端
    chan.get_pty()
    # 激活器
    chan.invoke_shell()
     
    while True:
        # 监视用户输入和服务器返回数据
        # sys.stdin 处理用户输入
        # chan 是之前创建的通道,用于接收服务器返回信息
        readable, writeable, error = select.select([chan, sys.stdin, ],[],[],1)
        if chan in readable:
            try:
                x = u(chan.recv(1024))
                if len(x) == 0:
                    print('
    *** EOF
    ')
                    break
                sys.stdout.write(x)
                sys.stdout.flush()
            except socket.timeout:
                pass
        if sys.stdin in readable:
            inp = sys.stdin.readline()
            chan.sendall(inp)
     
    chan.close()
    tran.close()
    
    import paramiko
    import sys
    import os
    import socket
    import select
    import getpass
    from paramiko.py3compat import u
    
    
    default_username = getpass.getuser()
    username = input('Username [%s]: ' % default_username)
    if len(username) == 0:
        username = default_username
    
    
    hostname = input('Hostname: ')
    if len(hostname) == 0:
        print('*** Hostname required.')
        sys.exit(1)
    
    tran = paramiko.Transport((hostname, 22,))
    tran.start_client()
    
    default_auth = "p"
    auth = input('Auth by (p)assword or (r)sa key[%s] ' % default_auth)
    if len(auth) == 0:
        auth = default_auth
    
    if auth == 'r':
        default_path = os.path.join(os.environ['HOME'], '.ssh', 'id_rsa')
        path = input('RSA key [%s]: ' % default_path)
        if len(path) == 0:
            path = default_path
        try:
            key = paramiko.RSAKey.from_private_key_file(path)
        except paramiko.PasswordRequiredException:
            password = getpass.getpass('RSA key password: ')
            key = paramiko.RSAKey.from_private_key_file(path, password)
        tran.auth_publickey(username, key)
    else:
        pw = getpass.getpass('Password for %s@%s: ' % (username, hostname))
        tran.auth_password(username, pw)
    
    
    
    # 打开一个通道
    chan = tran.open_session()
    # 获取一个终端
    chan.get_pty()
    # 激活器
    chan.invoke_shell()
    
    while True:
        # 监视用户输入和服务器返回数据
        # sys.stdin 处理用户输入
        # chan 是之前创建的通道,用于接收服务器返回信息
        readable, writeable, error = select.select([chan, sys.stdin, ],[],[],1)
        if chan in readable:
            try:
                x = u(chan.recv(1024))
                if len(x) == 0:
                    print('
    *** EOF
    ')
                    break
                sys.stdout.write(x)
                sys.stdout.flush()
            except socket.timeout:
                pass
        if sys.stdin in readable:
            inp = sys.stdin.readline()
            chan.sendall(inp)
    
    chan.close()
    tran.close()
    完整示例(一)

    3、肆意妄为(二)

    import paramiko
    import sys
    import os
    import socket
    import select
    import getpass
    import termios
    import tty
    from paramiko.py3compat import u
     
    tran = paramiko.Transport(('10.211.55.4', 22,))
    tran.start_client()
    tran.auth_password('wupeiqi', '123')
     
    # 打开一个通道
    chan = tran.open_session()
    # 获取一个终端
    chan.get_pty()
    # 激活器
    chan.invoke_shell()
     
     
    # 获取原tty属性
    oldtty = termios.tcgetattr(sys.stdin)
    try:
        # 为tty设置新属性
        # 默认当前tty设备属性:
        #   输入一行回车,执行
        #   CTRL+C 进程退出,遇到特殊字符,特殊处理。
     
        # 这是为原始模式,不认识所有特殊符号
        # 放置特殊字符应用在当前终端,如此设置,将所有的用户输入均发送到远程服务器
        tty.setraw(sys.stdin.fileno())
        chan.settimeout(0.0)
     
        while True:
            # 监视 用户输入 和 远程服务器返回数据(socket)
            # 阻塞,直到句柄可读
            r, w, e = select.select([chan, sys.stdin], [], [], 1)
            if chan in r:
                try:
                    x = u(chan.recv(1024))
                    if len(x) == 0:
                        print('
    *** EOF
    ')
                        break
                    sys.stdout.write(x)
                    sys.stdout.flush()
                except socket.timeout:
                    pass
            if sys.stdin in r:
                x = sys.stdin.read(1)
                if len(x) == 0:
                    break
                chan.send(x)
     
    finally:
        # 重新设置终端属性
        termios.tcsetattr(sys.stdin, termios.TCSADRAIN, oldtty)
     
     
    chan.close()
    tran.close()
    View Code
    import paramiko
    import sys
    import os
    import socket
    import getpass
    import termios
    import tty
    import select
    from paramiko.py3compat import u
    
    
    def interactive_shell(chan):
        # 获取原tty属性
        oldtty = termios.tcgetattr(sys.stdin)
        try:
            # 为tty设置新属性
            # 默认当前tty设备属性:
            #   输入一行回车,执行
            #   CTRL+C 进程退出,遇到特殊字符,特殊处理。
    
            # 这是为原始模式,不认识所有特殊符号
            # 放置特殊字符应用在当前终端,如此设置,将所有的用户输入均发送到远程服务器
            tty.setraw(sys.stdin.fileno())
            tty.setcbreak(sys.stdin.fileno())
            chan.settimeout(0.0)
            while True:
                r, w, e = select.select([chan, sys.stdin], [], [])
                if chan in r:
                    try:
                        x = u(chan.recv(1024))
                        if len(x) == 0:
                            sys.stdout.write('
    *** EOF
    ')
                            break
                        sys.stdout.write(x)
                        sys.stdout.flush()
                    except socket.timeout:
                        pass
                if sys.stdin in r:
                    x = sys.stdin.read(1)
                    if len(x) == 0:
                        break
                    chan.send(x)
    
        finally:
            # 重新设置终端属性
            termios.tcsetattr(sys.stdin, termios.TCSADRAIN, oldtty)
    
    
    def run():
        hostname = input('请输入主机名: ')
        tran = paramiko.Transport((hostname, 22,))
        tran.start_client()
    
        username = input('请输入用户名: ')
        auth = input('请输入密码进行验证(p) 或 (r)sa Key进行验证?')
        if auth == 'r':
            path = input('请输入RSA key 路径: ')
            try:
                key = paramiko.RSAKey.from_private_key_file(path)
            except paramiko.PasswordRequiredException:
                password = getpass.getpass('RSA key password: ')
                key = paramiko.RSAKey.from_private_key_file(path, password)
            tran.auth_publickey(username, key)
        else:
            pw = getpass.getpass('请输入密码 %s@%s: ' % (username, hostname))
            tran.auth_password(username, pw)
    
        # 打开一个通道
        chan = tran.open_session()
        # 获取一个终端
        chan.get_pty()
        # 激活器
        chan.invoke_shell()
    
        interactive_shell(chan)
    
        chan.close()
        tran.close()
    
    
    if __name__ == '__main__':
        run()
    基于Passwd或者RSA进行登陆操作
    import paramiko
    import sys
    import os
    import socket
    import getpass
    import termios
    import tty
    import select
    from paramiko.py3compat import u
    
    
    def interactive_shell(chan):
        # 获取原tty属性
        oldtty = termios.tcgetattr(sys.stdin)
        try:
            # 为tty设置新属性
            # 默认当前tty设备属性:
            # 输入一行回车,执行
            # CTRL+C 进程退出,遇到特殊字符,特殊处理。
    
            # 这是为原始模式,不认识所有特殊符号
            # 放置特殊字符应用在当前终端,如此设置,将所有的用户输入均发送到远程服务器
            tty.setraw(sys.stdin.fileno())
            tty.setcbreak(sys.stdin.fileno())
            chan.settimeout(0.0)
            while True:
                r, w, e = select.select([chan, sys.stdin], [], [])
                if chan in r:
                    try:
                        x = u(chan.recv(1024))
                        if len(x) == 0:
                            sys.stdout.write('
    *** EOF
    ')
                            break
                        sys.stdout.write(x)
                        sys.stdout.flush()
                    except socket.timeout:
                        pass
                if sys.stdin in r:
                    x = sys.stdin.read(1)
                    if len(x) == 0:
                        break
                    chan.send(x)
    
        finally:
            # 重新设置终端属性
            termios.tcsetattr(sys.stdin, termios.TCSADRAIN, oldtty)
    
    
    def run():
        db_dict = {
            'c1.salt.com': {
                'root': {'user': 'root', 'auth': 'r', "cert": 'key路径'},
                'alex': {'user': 'alex', 'auth': 'p', "cert": '密码'},
            },
            'c2.salt.com': {
                'alex': {'user': 'alex', 'auth': 'p', "cert": '密码'},
            },
    
        }
    
        for row in db_dict.keys():
            print(row)
    
        hostname = input('请选择主机: ')
        tran = paramiko.Transport((hostname, 22,))
        tran.start_client()
    
        for item in db_dict[hostname].keys():
            print(item)
    
        username = input('请输入用户: ')
    
        user_dict = db_dict[hostname][username]
        if username['auth'] == 'r':
            key = paramiko.RSAKey.from_private_key_file(user_dict['cert'])
            tran.auth_publickey(username, key)
        else:
            pw = user_dict['cert']
            tran.auth_password(username, pw)
    
        # 打开一个通道
        chan = tran.open_session()
        # 获取一个终端
        chan.get_pty()
        # 激活器
        chan.invoke_shell()
    
        interactive_shell(chan)
    
        chan.close()
        tran.close()
    
    
    if __name__ == '__main__':
        run()
    提示用户选择主机和用户
    import paramiko
    import sys
    import os
    import socket
    import getpass
    import termios
    import tty
    import select
    from paramiko.py3compat import u
    
    
    def interactive_shell(chan):
        # 获取原tty属性
        oldtty = termios.tcgetattr(sys.stdin)
        try:
            # 为tty设置新属性
            # 默认当前tty设备属性:
            # 输入一行回车,执行
            # CTRL+C 进程退出,遇到特殊字符,特殊处理。
    
            # 这是为原始模式,不认识所有特殊符号
            # 放置特殊字符应用在当前终端,如此设置,将所有的用户输入均发送到远程服务器
            tty.setraw(sys.stdin.fileno())
            tty.setcbreak(sys.stdin.fileno())
            chan.settimeout(0.0)
    
            log = open('handle.log', 'a+', encoding='utf-8')
            flag = False
            temp_list = []
    
            while True:
                r, w, e = select.select([chan, sys.stdin], [], [])
                if chan in r:
                    try:
                        x = u(chan.recv(1024))
                        if len(x) == 0:
                            sys.stdout.write('
    *** EOF
    ')
                            break
                        # 如果用户上一次点击的是tab键,则获取返回的内容写入在记录中
                        if flag:
                            if x.startswith('
    '):
                                pass
                            else:
                                temp_list.append(x)
                            flag = False
                        sys.stdout.write(x)
                        sys.stdout.flush()
                    except socket.timeout:
                        pass
                if sys.stdin in r:
                    # 读取用户在终端数据每一个字符
                    x = sys.stdin.read(1)
                    if len(x) == 0:
                        break
                    # 如果用户点击TAB键
                    if x == '	':
                        flag = True
                    else:
                        # 未点击TAB键,则将每个操作字符记录添加到列表中,以便之后写入文件
                        temp_list.append(x)
    
                    # 如果用户敲回车,则将操作记录写入文件
                    if x == '
    ':
                        log.write(''.join(temp_list))
                        log.flush()
                        temp_list.clear()
                    chan.send(x)
    
        finally:
            # 重新设置终端属性
            termios.tcsetattr(sys.stdin, termios.TCSADRAIN, oldtty)
    
    
    def run():
        db_dict = {
            'c1.salt.com': {
                'root': {'user': 'root', 'auth': 'r', "cert": 'key路径'},
                'alex': {'user': 'alex', 'auth': 'p', "cert": '密码'},
            },
            'c2.salt.com': {
                'root': {'user': 'root', 'auth': 'r', "cert": 'key路径'},
                'alex': {'user': 'alex', 'auth': 'p', "cert": '密码'},
            },
    
        }
    
        for row in db_dict.keys():
            print(row)
    
        hostname = input('请选择主机: ')
        tran = paramiko.Transport((hostname, 22,))
        tran.start_client()
    
        for item in db_dict[hostname].keys():
            print(item)
    
        username = input('请输入用户: ')
    
        user_dict = db_dict[hostname][username]
        if username['auth'] == 'r':
            key = paramiko.RSAKey.from_private_key_file(user_dict['cert'])
            tran.auth_publickey(username, key)
        else:
            pw = user_dict['cert']
            tran.auth_password(username, pw)
    
        # 打开一个通道
        chan = tran.open_session()
        # 获取一个终端
        chan.get_pty()
        # 激活器
        chan.invoke_shell()
    
        interactive_shell(chan)
    
        chan.close()
        tran.close()
    
    
    if __name__ == '__main__':
        run()
    提示用户选择主机和用户(记录操作日志)

  • 相关阅读:
    发送指令
    WIN32得到HWND
    查找摄像头
    WindowImplBase::OnSysCommand-------duilib在最大化和还原间切换
    CImage将图片转为指定像素大小
    聚集索引和非聚集索引(整理)
    数据库SQL优化大总结之 百万级数据库优化方案
    架构师之路(39)---IoC框架
    .NET Reflector反编译的方法
    PowerDesigner之PDM(物理概念模型)
  • 原文地址:https://www.cnblogs.com/charliedaifu/p/10098848.html
Copyright © 2020-2023  润新知