“轮询”指的是什么,此代码到底在做什么?

时间:2019-05-29 19:56:14

标签: python networking ssh paramiko polling

我目前正在构建一个python脚本,该脚本可启用和禁用交换机上的某些端口,以查看交换机将如何工作以达到质量保证目的。我一直在使用一个名为paramiko的Python库,该库实现SSH以连接到我想要的任何设备,我指的是我的一个队友分发的旧代码来编写更多脚本。在一个旧版代码文件中,有一个名为_run_poll的函数,我不明白它的确切作用。

我已经尝试对“轮询”在SSH方面的含义进行一些研究,但是我仍然不了解运行“轮询”时的情况。它的定义似乎有点模糊。这是函数:

def _run_poll(self, session, timeout, input_data):

    interval = 0.1
    maxseconds = timeout
    maxcount = maxseconds / interval


    i = 0
    timeout_flag = False
    self.info('polling (%d, %d)' % (maxseconds, maxcount))
    start = datetime.datetime.now()
    start_secs = time.mktime(start.timetuple())
    output = ''
    session.setblocking(0)
    while True:
        if session.recv_ready(): # returns true if data has been buffered
            data = session.recv(self.bufsize) # receive data from the channel
            output += data
            self.info('read %d bytes, total %d' % (len(data), len(output)))

            if session.send_ready(): 
                # We received a potential prompt.
                # In the future this could be made to work more like
                # pexpect with pattern matching.
                if i < len(input_data):
                    data = input_data[input_idx] + '\n'
                    i += 1
                    self.info('sending input data %d' % (len(data)))
                    session.send(data)

        self.info('session.exit_status_ready() = %s' % (str(session.exit_status_ready())))
        if session.exit_status_ready():
            break

        # Timeout check
        now = datetime.datetime.now()
        now_secs = time.mktime(now.timetuple())
        et_secs = now_secs - start_secs
        self.info('timeout check %d %d' % (et_secs, maxseconds))
        if et_secs > maxseconds:
            self.info('polling finished - timeout')
            timeout_flag = True
            break
        time.sleep(0.200)

    self.info('polling loop ended')
    if session.recv_ready():
        data = session.recv(self.bufsize)
        output += data
        self.info('read %d bytes, total %d' % (len(data), len(output)))

    self.info('polling finished - %d output bytes' % (len(output)))
    if timeout_flag:
        self.info('appending timeout message')
        output += '\nERROR: timeout after %d seconds\n' % (timeout)
        session.close()

    return output

我找不到很多在线资源来描述这里发生的事情或有关“轮询”的一般信息。谁能帮我解释一下“轮询”到底是什么,这是怎么回事?

2 个答案:

答案 0 :(得分:1)

在编程中有两种处理异步事件的方法。

一种方法是使用中断:您的代码只有在某种机制“唤醒”后才执行,然后执行。必须在比您的代码执行位置更低的级别上支持此机制。例如,微控制器具有内置的特定硬件,这些硬件可以中断应用程序并跳转到特定地址以开始执行指令以处理中断。

构建中断系统非常困难,并且需要大量工作。对于某些应用程序,这是不可能的。轮询或重复检查条件直到其变为True几乎总是比较容易(尽管效率较低),然后继续执行其他操作。

在您的示例中,请注意他如何使用while True:循环。 True永远不会为False,因此while循环只能由break语句来打断。我们在以下位置找到了break语句

if session.exit_status_ready():
            break

因此,该代码的编写者决定继续做某事,直到session.exit_status_ready()为True。由于这是paramiko,很可能他通过SSH执行了远程命令,并且正在等待命令完成并返回退出代码。该循环的目的是使程序停留在循环中,直到命令执行完毕并返回结果为止。它还可能会超时:

if et_secs > maxseconds:
            self.info('polling finished - timeout')
            timeout_flag = True
            break

因此,如果命令花费的时间超过maxseconds,则该程序将不会永远坐着等待。

退出循环后,它会打印:

self.info('polling loop ended')

因此,当您看到此消息时,便知道远程命令已完成执行或超时。

轮询的重点是反复检查某些事物,直到出现特定情况为止。在您的情况下,该条件是“远程命令已完成执行”或“已过去一定时间”。

答案 1 :(得分:0)

关于Polling的维基百科文章将其定义为:

  

作为同步活动,通过客户端程序主动采样外部设备的状态。


在您的代码中,这意味着它会定期(每200毫秒)检查SSH连接是否有任何传入数据以及输出队列中的可用容量。