| # Copyright 2022 OpenStack Foundation |
| # All Rights Reserved. |
| # |
| # Licensed under the Apache License, Version 2.0 (the "License"); you may |
| # not use this file except in compliance with the License. You may obtain |
| # a copy of the License at |
| # |
| # http://www.apache.org/licenses/LICENSE-2.0 |
| # |
| # Unless required by applicable law or agreed to in writing, software |
| # distributed under the License is distributed on an "AS IS" BASIS, WITHOUT |
| # WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the |
| # License for the specific language governing permissions and limitations |
| # under the License. |
| import signal |
| import subprocess |
| |
| import fixtures |
| |
| from oslo_log import log |
| |
| |
| LOG = log.getLogger(__name__) |
| |
| |
| class NetDowntimeMeter(fixtures.Fixture): |
| def __init__(self, dest_ip, interval='0.2'): |
| self.dest_ip = dest_ip |
| # Note: for intervals lower than 0.2 ping requires root privileges |
| self.interval = interval |
| self.ping_process = None |
| |
| def _setUp(self): |
| self.start_background_pinger() |
| |
| def start_background_pinger(self): |
| cmd = ['ping', '-q', '-s1'] |
| cmd.append('-i{}'.format(self.interval)) |
| cmd.append(self.dest_ip) |
| LOG.debug("Starting background pinger to '{}' with interval {}".format( |
| self.dest_ip, self.interval)) |
| self.ping_process = subprocess.Popen( |
| cmd, stdout=subprocess.PIPE, stderr=subprocess.PIPE) |
| self.addCleanup(self.cleanup) |
| |
| def cleanup(self): |
| if self.ping_process and self.ping_process.poll() is None: |
| LOG.debug('Terminating background pinger with pid {}'.format( |
| self.ping_process.pid)) |
| self.ping_process.terminate() |
| self.ping_process = None |
| |
| def get_downtime(self): |
| self.ping_process.send_signal(signal.SIGQUIT) |
| # Example of the expected output: |
| # 264/274 packets, 3% loss |
| output = self.ping_process.stderr.readline().strip().decode('utf-8') |
| if output and len(output.split()[0].split('/')) == 2: |
| succ, total = output.split()[0].split('/') |
| return (int(total) - int(succ)) * float(self.interval) |
| else: |
| LOG.warning('Unexpected output obtained from the pinger: %s', |
| output) |