使用python库paramiko实现ssh登录并通过tunnels转发

由于工作需要,需要写个端口转发的工具,google了一下发现大多数都用paramiko库的比较多,并且github上也有一个脚本的示例,非常好用。
脚本如下:

import getpass
import os
import socket
import select
import json

try:
    import SocketServer
except ImportError:
    import socketserver as SocketServer

import sys
from optparse import OptionParser

import paramiko

SSH_PORT = 2637
DEFAULT_PORT = 4151

g_verbose = True


class ForwardServer(SocketServer.ThreadingTCPServer):
    daemon_threads = True
    allow_reuse_address = True


class Handler(SocketServer.BaseRequestHandler):

    def handle(self):
        try:
            chan = self.ssh_transport.open_channel(
                "direct-tcpip",
                (self.chain_host, self.chain_port),
                self.request.getpeername(),
            )
        except Exception as e:
            verbose(
                "Incoming request to %s:%d failed: %s"
                % (self.chain_host, self.chain_port, repr(e))
            )
            return
        if chan is None:
            verbose(
                "Incoming request to %s:%d was rejected by the SSH server."
                % (self.chain_host, self.chain_port)
            )
            return

        while True:
            r, w, x = select.select([self.request, chan], [], [])
            if self.request in r:
                data = self.request.recv(1024)
                if len(data) == 0:
                    break
                chan.send(data)
            if chan in r:
                data = chan.recv(1024)
                if len(data) == 0:
                    break
                self.request.send(data)

        peername = self.request.getpeername()
        chan.close()
        self.request.close()


def forward_tunnel(local_port, remote_host, remote_port, transport):
    class SubHander(Handler):
        chain_host = remote_host
        chain_port = remote_port
        ssh_transport = transport

    ForwardServer(("", local_port), SubHander).serve_forever()


def verbose(s):
    if g_verbose:
        print(s)


HELP = """\
Set up a forward tunnel across an SSH server, using paramiko. A local port
(given with -p) is forwarded across an SSH session to an address:port from
the SSH server. This is similar to the openssh -L option.
"""


def get_host_port(spec, default_port):
    "parse 'hostname:22' into a host and port, with the port optional"
    args = (spec.split(":", 1) + [default_port])[:2]
    args[1] = int(args[1])
    return args[0], args[1]


def parse_options():
    global g_verbose

    parser = OptionParser(
        usage="usage: python update.py 111111.test",
        version="%prog 1.0",
        description=HELP,
    )
    parser.add_option(
        "-q",
        "--quiet",
        action="store_false",
        dest="verbose",
        default=True,
        help="squelch all informational output",
    )
    parser.add_option(
        "-p",
        "--local-port",
        action="store",
        type="int",
        dest="port",
        default=DEFAULT_PORT,
        help="local port to forward (default: %d)" % DEFAULT_PORT,
    )
    parser.add_option(
        "-u",
        "--user",
        action="store",
        type="string",
        dest="user",
        default='hex',
        help="username for SSH authentication (default: %s)"
             % getpass.getuser(),
    )
    parser.add_option(
        "-K",
        "--key",
        action="store",
        type="string",
        dest="keyfile",
        default=None,
        help="private key file to use for SSH authentication",
    )
    parser.add_option(
        "",
        "--no-key",
        action="store_false",
        dest="look_for_keys",
        default=True,
        help="don't look for or use a private key file",
    )
    parser.add_option(
        "-P",
        "--password",
        action="store",
        dest="readpass",
        default='xxxxx',
        help="read password (for key or password auth) from stdin",
    )
    parser.add_option(
        "-r",
        "--remote",
        action="store",
        type="string",
        dest="remote",
        default='192.168.2.159:1111',
        metavar="host:port",
        help="remote host and port to forward to",
    )
    parser.add_option(
        "-s",
        "--ssh-host",
        action="store",
        type="string",
        dest="ssh_host",
        default='222.22.22.22:1111',
        metavar="host:port",
        help="ssh host and port",
    )
    parser.add_option(
        "-d",
        "--device-type",
        action="store",
        type="string",
        dest="device_type",
        default='device',
        help="group or device",
    )
    parser.add_option(
        "-i",
        "--device-ids",
        action="store",
        type="string",
        dest="device_ids",
        default=None,
        help="deveice ids",
    )
    parser.add_option(
        "-w",
        "--wait-second",
        action="store",
        type="int",
        dest="wait_second",
        default=120,
        help="wait_second",
    )

    options, args = parser.parse_args()

    if options.remote is None:
        parser.error("Remote address required (-r).")

    g_verbose = options.verbose
    server_host, server_port = get_host_port(options.ssh_host, SSH_PORT)
    remote_host, remote_port = get_host_port(options.remote, SSH_PORT)
    return options, (server_host, server_port), (remote_host, remote_port), args, parser


def main():
    options, server, remote, args, parser = parse_options()

    password = options.readpass

    client = paramiko.SSHClient()
    client.load_system_host_keys()
    client.set_missing_host_key_policy(paramiko.WarningPolicy())

    verbose("Connecting to ssh host %s:%d ..." % (server[0], server[1]))
    try:
        client.connect(
            server[0],
            server[1],
            username=options.user,
            key_filename=options.keyfile,
            look_for_keys=options.look_for_keys,
            password=password,
        )
    except Exception as e:
        print("*** Failed to connect to %s:%d: %r" % (server[0], server[1], e))
        sys.exit(1)

    verbose(
        "Now forwarding port %d to %s:%d ..."
        % (options.port, remote[0], remote[1])
    )
    print('\n\n\n\n')
    try:
        from threading import Thread
        import time
        import requests
        import json
        tunnel_thread = Thread(target=forward_tunnel, args=(options.port, remote[0],
                                                            remote[1],
                                                            client.get_transport()))
        tunnel_thread.start()
        if len(args) != 2:
            print(args)
            parser.error("Incorrect number of arguments. len=" + str(len(args)))
        topic = args[0]
        hipos_zip = args[1]
        if not hipos_zip:
            parser.error("Need hipos_zip argument.")
        if not topic:
            parser.error("Need topic argument.")
        if topic == '1201.HEX_POS_SERVER_DEVICE':
            print("Online topic:%s %s" % (topic, '*' * 60))
            online_flag = input("It's online topic, please enter 'true' for sure***************:  ")
            if 'true' != online_flag:
                parser.error("Not continue! It's online topic.")
                os._exit(0)
        elif topic == '120101.HEX_POS_SERVER_DEVICE':
            print('Test topic:%s %s' % (topic, '*' * 60))
        else:
            parser.error("Invalid topic argument.")
        device_type = options.device_type
        ids = options.device_ids
        if not ids or not ids.split(','):
            parser.error("Please enter ids!")
        url = "http://localhost:4151/pub?topic=" + topic
        config = ids.split(',')
        update_data = {
            'event':
                'local_api',
            'type':
                'device',
            'group':
                '',
            'id':
                'nothing',
            'payload': {
                'data':
                    [None,
                     'hipos_update.zip',
                     '13953192',
                     'temp'],
                'action':
                    'updateToNewHipos',
                'id':
                    0,
                'serviceName':
                    'version'}
        }
        update_data['type'] = device_type
        update_data['payload']['data'][0] = hipos_zip
        for i in range(1, 61):
            time.sleep(1)
            print('Download update_zip command will be sent in %ss, please wait...' %
                  (61 - i))
        for i in config:
            update_data['id'] = i
            print('id=' + str(update_data['id']))
            print('update=' + str(update_data))
            resp = requests.post(url, data=json.dumps(update_data))
            print('response=' + resp.text)
            time.sleep(1)
        wait_second = options.wait_second
        
        print('Send update data successfully, restart in %s s.' % wait_second)
        print('Please waiting')
        for i in range(1, wait_second):
            time.sleep(1)
            print('Restart command will be sent in %ss, please wait...' %
                  (wait_second - i))
        update_data['payload']['data'] = []
        update_data['payload']['action'] = 'RestartToNewHipos'
        print('update=' + str(update_data))
        for i in config:
            update_data['id'] = i
            print('Restart command was sent successfully! ID=' + str(i))
            print('update=' + str(update_data))
            resp = requests.post(url, data=json.dumps(update_data))
            print('response=' + resp.text)
            time.sleep(1)
        print('Wait for hipos coming')
        print('done!')
        os._exit(0)
    except KeyboardInterrupt:
        print("C-c: Port forwarding stopped.")
        os._exit(0)
    return client


if __name__ == "__main__":
    main()
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 5
    评论
使用Python的paramiko库可以实现SSH远程登录H3C核心交换机,具体步骤如下: 1. 安装paramiko库,可以使用pip命令进行安装: ``` pip install paramiko ``` 2. 导入paramiko库: ```python import paramiko ``` 3. 设置SSH登录信息: ```python host = '192.168.1.1' # H3C交换机的IP地址 username = 'admin' # 登录用户名 password = 'password' # 登录密码 ``` 4. 创建SSH客户端,并进行SSH连接: ```python client = paramiko.SSHClient() client.set_missing_host_key_policy(paramiko.AutoAddPolicy()) client.connect(hostname=host, username=username, password=password) ``` 其中,`set_missing_host_key_policy`方法用于设置SSH连接时自动添加缺失的主机密钥,`connect`方法用于进行SSH连接。 5. 执行命令: ```python stdin, stdout, stderr = client.exec_command('display version') result = stdout.read().decode('utf-8') ``` 在SSH连接成功后,可以使用`exec_command`方法执行命令,并从返回值中获取命令的执行结果。 6. 关闭SSH连接: ```python client.close() ``` 完整代码示例: ```python import paramiko host = '192.168.1.1' username = 'admin' password = 'password' client = paramiko.SSHClient() client.set_missing_host_key_policy(paramiko.AutoAddPolicy()) client.connect(hostname=host, username=username, password=password) stdin, stdout, stderr = client.exec_command('display version') result = stdout.read().decode('utf-8') print(result) client.close() ``` 在使用paramiko库进行SSH远程登录时,需要注意H3C交换机是否开启了SSH服务,并设置了正确的登录信息,另外还需要注意交换机的配置不要出现问题,以免造成不必要的故障。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值