Python的SSH隧道自动关闭

时间:2016-10-07 15:56:58

标签: python mysql python-3.x ssh

我需要一些关于程序结构的建议。我正在使用sshtunnel连接到外部MySQL数据库。它现在正常工作(我可以发出SQL命令并获得结果),但前提是这些命令与连接的开放功能相同。如果它们处于不同的功能,隧道将自动关闭,然后才能使用它。 (见下面的代码 - 它在两个检查点之间关闭。)所以我的问题是:

  1. 我每次要使用它时都应该检查连接是否打开?我该怎么做?
  2. 如何使用其他功能的连接?我在sshtunnel中看到了一个名为“keepalive”的属性(这将使连接保持打开指定的时间长度) - 这是我需要的吗?我该如何使用它?
  3. 我可以忘记手动关闭隧道吗?
  4. 还有什么能帮我搞定这个吗?你可能会说,我是这个主题的新手!
  5. 感谢。

    Python脚本:

    import pymysql, shlex, shutil, subprocess
    import logging
    import sshtunnel
    from sshtunnel import SSHTunnelForwarder
    import iot_config as cfg
    
    def OpenRemoteDB():
        global remotecur, remotedb
        sshtunnel.DEFAULT_LOGLEVEL = logging.DEBUG
        with SSHTunnelForwarder(
                (cfg.sshconn['host'], cfg.sshconn['port']),
                ssh_username = cfg.sshconn['user'],
                ssh_private_key = cfg.sshconn['private_key_loc'],
                ssh_private_key_password = cfg.sshconn['private_key_passwd'],
                remote_bind_address = ('127.0.0.1', 3306)) as server:
            remotedb = None
            remotedb = pymysql.connect(host='127.0.0.1', user=cfg.remotedbconn['user'], passwd=cfg.remotedbconn['passwd'], db=cfg.remotedbconn['db'], port=server.local_bind_port)
            remotecur = remotedb.cursor()
            print("Checkpoint 1")
            #The next three lines work fine
    #        remotecur.execute("SELECT ActionID, Description FROM cmAction")
    #        for r in remotecur:
    #            print(r)
    
    def SyncActions():
        print("Checkpoint 2")
        #the next three lines don't work (because the connection has closed)
        remotecur.execute("SELECT ActionID, Description FROM cmAction")
        for r in remotecur:
            print(r)
    
    # Main program starts here
    OpenRemoteDB()
    SyncActions()
    

    输出:

    2016-10-06 12:34:21,088| WAR | MainThrea/0954@sshtunnel | Could not read SSH configuration file: ~/.ssh/config
    2016-10-06 12:34:21,153| INF | MainThrea/0981@sshtunnel | 0 keys loaded from agent
    2016-10-06 12:34:21,963| DEB | MainThrea/1160@sshtunnel | Private key file (/etc/ssh/lavenham_private_key.key, <class 'paramiko.rsakey.RSAKey'>) successfully loaded
    2016-10-06 12:34:22,003| INF | MainThrea/0901@sshtunnel | Connecting to gateway: lavenham.co.uk:22 as user 'lavenham'
    2016-10-06 12:34:22,062| DEB | MainThrea/0904@sshtunnel | Concurrent connections allowed: True
    2016-10-06 12:34:22,117| DEB | MainThrea/1300@sshtunnel | Trying to log in with key: b'611711d06f2b671960c3458d25ca3c20'
    2016-10-06 12:34:23,083| INF | Srv-39033/1334@sshtunnel | Opening tunnel: 0.0.0.0:39033 <> 127.0.0.1:3306
    Checkpoint 1
    2016-10-06 12:34:23,290| INF | MainThrea/1350@sshtunnel | Shutting down tunnel ('0.0.0.0', 39033)
    2016-10-06 12:34:23,424| INF | Srv-39033/1340@sshtunnel | Tunnel: 0.0.0.0:39033 <> 127.0.0.1:3306 released
    2016-10-06 12:34:23,426| DEB | MainThrea/1363@sshtunnel | Transport is closed
    Checkpoint 2
    Traceback (most recent call last):
      File "/home/pi/Documents/iot_pm2/2016-10-06.py", line 33, in <module>
        SyncActions()
      File "/home/pi/Documents/iot_pm2/2016-10-06.py", line 27, in SyncActions
        remotecur.execute("SELECT ActionID, Description FROM cmAction")
      File "/usr/local/lib/python3.4/dist-packages/pymysql/cursors.py", line 146, in execute
        result = self._query(query)
      File "/usr/local/lib/python3.4/dist-packages/pymysql/cursors.py", line 296, in _query
        conn.query(q)
      File "/usr/local/lib/python3.4/dist-packages/pymysql/connections.py", line 819, in query
        self._affected_rows = self._read_query_result(unbuffered=unbuffered)
      File "/usr/local/lib/python3.4/dist-packages/pymysql/connections.py", line 1001, in _read_query_result
        result.read()
      File "/usr/local/lib/python3.4/dist-packages/pymysql/connections.py", line 1285, in read
        first_packet = self.connection._read_packet()
      File "/usr/local/lib/python3.4/dist-packages/pymysql/connections.py", line 945, in _read_packet
        packet_header = self._read_bytes(4)
      File "/usr/local/lib/python3.4/dist-packages/pymysql/connections.py", line 981, in _read_bytes
        2013, "Lost connection to MySQL server during query")
    pymysql.err.OperationalError: (2013, 'Lost connection to MySQL server during query')
    

1 个答案:

答案 0 :(得分:0)

根据上面steven-rumbalski的评论:

替换: with SSHTunnelForwarder(...) as server
使用: server = SSHTunnelForwarder(...)
然后包裹server.start() ... server.stop()
围绕要通过SSH隧道发送的代码。

这是转换后的代码:

import pymysql, shlex, shutil, subprocess
import logging
import sshtunnel
from sshtunnel import SSHTunnelForwarder
import iot_config as cfg

def OpenSSHTunnel():
    global server
    sshtunnel.DEFAULT_LOGLEVEL = logging.DEBUG
    server = SSHTunnelForwarder(
        (cfg.sshconn['host'], cfg.sshconn['port']),
        ssh_username = cfg.sshconn['user'],
        ssh_private_key = cfg.sshconn['private_key_loc'],
        ssh_private_key_password = cfg.sshconn['private_key_passwd'],
        remote_bind_address = ('127.0.0.1', 3306)
    )

def OpenRemoteDB():
    global remotecur, remotedb
    remotedb = None
    remotedb = pymysql.connect(
        host='127.0.0.1',
        user=cfg.remotedbconn['user'],
        passwd=cfg.remotedbconn['passwd'],
        db=cfg.remotedbconn['db'],
        port=server.local_bind_port
    )
    remotecur = remotedb.cursor()
    print("Checkpoint 1")

def SyncActions():
    print("Checkpoint 2")
    # this should now work as expected
    remotecur.execute("SELECT ActionID, Description FROM cmAction")
    for r in remotecur:
        print(r)

# Main program starts here
OpenSSHTunnel()
server.start()
OpenRemoteDB()
SyncActions()
server.stop()