Replace tobiko.common.utils.network with tobiko.shell.ping
Change-Id: I55504fd6387593297f671b76753410d6414ef6b1
This commit is contained in:
parent
f26cb5c276
commit
081ff5240c
@ -2,6 +2,5 @@
|
|||||||
|
|
||||||
ansible>=2.4.0 # GPLv3
|
ansible>=2.4.0 # GPLv3
|
||||||
os-faults>=0.1.18 # Apache-2.0
|
os-faults>=0.1.18 # Apache-2.0
|
||||||
tempest>=17.1.0 # Apache-2.0
|
|
||||||
cryptography<=2.2.2 # Apache-2.0
|
cryptography<=2.2.2 # Apache-2.0
|
||||||
Jinja2>=2.8.0 # BSD
|
Jinja2>=2.8.0 # BSD
|
||||||
|
@ -1,141 +0,0 @@
|
|||||||
# Copyright 2018 Red Hat
|
|
||||||
#
|
|
||||||
# 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.
|
|
||||||
from __future__ import absolute_import
|
|
||||||
|
|
||||||
import glob
|
|
||||||
import inspect
|
|
||||||
import os
|
|
||||||
import re
|
|
||||||
import signal
|
|
||||||
import subprocess
|
|
||||||
|
|
||||||
from oslo_log import log
|
|
||||||
from tempest.common.utils import net_utils
|
|
||||||
from tempest.lib.common.utils import test_utils
|
|
||||||
|
|
||||||
import tobiko
|
|
||||||
from tobiko import config
|
|
||||||
|
|
||||||
LOG = log.getLogger(__name__)
|
|
||||||
SHELL = (config.get_any_option('tobiko.shell.command') or '/bin/sh -c').split()
|
|
||||||
|
|
||||||
|
|
||||||
SG_RULES = {'ALLOW_ICMP':
|
|
||||||
{'direction': 'ingress',
|
|
||||||
'protocol': 'icmp'
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
def run_background_ping(ip):
|
|
||||||
"""Starts background ping process."""
|
|
||||||
# The caller function name
|
|
||||||
caller_f = inspect.stack()[1][1].split('/')[-1].split(".py")[0]
|
|
||||||
PING_OUTPUT_F = "/tmp/ping_%s_%s_output" % (ip, caller_f)
|
|
||||||
PING_PID_F = "/tmp/ping_%s_%s_pid" % (ip, caller_f)
|
|
||||||
PING_CMD = 'ping -q %s' % ip
|
|
||||||
|
|
||||||
# Kill any existing running ping process to the same IP address
|
|
||||||
if os.path.exists(PING_OUTPUT_F):
|
|
||||||
with os.open(PING_OUTPUT_F) as pid_file:
|
|
||||||
os.kill(int(pid_file.read()), signal.SIGINT)
|
|
||||||
for f in glob.glob("/tmp/ping_%s_%s*" % (ip, caller_f)):
|
|
||||||
os.remove(f)
|
|
||||||
for f in glob.glob("/tmp/ping_%s_%s.*"):
|
|
||||||
os.remove(f)
|
|
||||||
|
|
||||||
ping_log = open(PING_OUTPUT_F, 'ab')
|
|
||||||
p = subprocess.Popen([PING_CMD], stdout=ping_log, shell=True)
|
|
||||||
with open(PING_PID_F, 'ab') as pf:
|
|
||||||
pf.write(str(p.pid))
|
|
||||||
|
|
||||||
|
|
||||||
def get_packet_loss(ip):
|
|
||||||
"""Returns packet loss."""
|
|
||||||
|
|
||||||
# The caller function name
|
|
||||||
caller_f = inspect.stack()[1][1].split('/')[-1].split(".py")[0]
|
|
||||||
PING_OUTPUT_F = "/tmp/ping_%s_%s_output" % (ip, caller_f)
|
|
||||||
PING_PID_F = "/tmp/ping_%s_%s_pid" % (ip, caller_f)
|
|
||||||
|
|
||||||
try:
|
|
||||||
# Kill Process
|
|
||||||
with open(PING_PID_F) as f:
|
|
||||||
pid = f.read()
|
|
||||||
os.kill(int(pid), signal.SIGINT)
|
|
||||||
|
|
||||||
# Packet loss pattern
|
|
||||||
p = re.compile("(\\d{1,3})% packet loss")
|
|
||||||
|
|
||||||
# Get ping package loss
|
|
||||||
with open(PING_OUTPUT_F) as f:
|
|
||||||
m = p.search(f.read())
|
|
||||||
packet_loss = m.group(1)
|
|
||||||
finally:
|
|
||||||
# Remove files created by pre test
|
|
||||||
|
|
||||||
from shutil import copyfile
|
|
||||||
copyfile(PING_OUTPUT_F, "/home/abregman/stam")
|
|
||||||
os.remove(PING_OUTPUT_F)
|
|
||||||
os.remove(PING_PID_F)
|
|
||||||
|
|
||||||
return packet_loss
|
|
||||||
|
|
||||||
|
|
||||||
def ping_ip_address(ip_address, should_succeed=True,
|
|
||||||
ping_timeout=None, mtu=None, fragmentation=True):
|
|
||||||
if not ip_address:
|
|
||||||
raise ValueError("Invalid IP address: {!r}".format(ip_address))
|
|
||||||
|
|
||||||
timeout = ping_timeout or 60.
|
|
||||||
cmd = ['ping', '-c1', '-w10']
|
|
||||||
|
|
||||||
if mtu:
|
|
||||||
if not fragmentation:
|
|
||||||
cmd += ['-M', 'do']
|
|
||||||
cmd += ['-s', str(net_utils.get_ping_payload_size(mtu, 4))]
|
|
||||||
cmd.append(ip_address)
|
|
||||||
cmd_line = SHELL + [str(subprocess.list2cmdline(cmd))]
|
|
||||||
|
|
||||||
def ping():
|
|
||||||
LOG.debug('Execute ping command: %r', cmd_line)
|
|
||||||
proc = subprocess.Popen(cmd_line,
|
|
||||||
stdout=subprocess.PIPE,
|
|
||||||
stderr=subprocess.PIPE,
|
|
||||||
universal_newlines=True)
|
|
||||||
stdout, stderr = proc.communicate()
|
|
||||||
if proc.returncode == 0:
|
|
||||||
LOG.debug("Ping command succeeded:\n"
|
|
||||||
"stdout:\n%s\n"
|
|
||||||
"stderr:\n%s\n", stdout, stderr)
|
|
||||||
return should_succeed
|
|
||||||
else:
|
|
||||||
LOG.debug("Ping command failed (exit_status=%d):\n"
|
|
||||||
"stdout:\n%s\n"
|
|
||||||
"stderr:\n%s\n", proc.returncode, stdout, stderr)
|
|
||||||
return not should_succeed
|
|
||||||
|
|
||||||
return test_utils.call_until_true(ping, timeout, 1)
|
|
||||||
|
|
||||||
|
|
||||||
def assert_ping(ip, should_fail=False, fragmentation=True,
|
|
||||||
packet_size=None):
|
|
||||||
success = ping_ip_address(ip, mtu=packet_size,
|
|
||||||
fragmentation=fragmentation)
|
|
||||||
if success:
|
|
||||||
if should_fail:
|
|
||||||
tobiko.fail("Host {!r} is reachable", ip)
|
|
||||||
|
|
||||||
elif not should_fail:
|
|
||||||
tobiko.fail("Host {!r} is not reachable", ip)
|
|
@ -37,7 +37,7 @@ RECEIVED = 'received'
|
|||||||
UNRECEIVED = 'unreceived'
|
UNRECEIVED = 'unreceived'
|
||||||
|
|
||||||
|
|
||||||
def ping(host, until=TRANSMITTED, **ping_params):
|
def ping(host, until=TRANSMITTED, check=True, **ping_params):
|
||||||
"""Send ICMP messages to host address until timeout
|
"""Send ICMP messages to host address until timeout
|
||||||
|
|
||||||
:param host: destination host address
|
:param host: destination host address
|
||||||
@ -45,7 +45,7 @@ def ping(host, until=TRANSMITTED, **ping_params):
|
|||||||
function
|
function
|
||||||
:returns: PingStatistics
|
:returns: PingStatistics
|
||||||
"""
|
"""
|
||||||
return get_statistics(host=host, until=until, **ping_params)
|
return get_statistics(host=host, until=until, check=check, **ping_params)
|
||||||
|
|
||||||
|
|
||||||
def ping_until_delivered(host, **ping_params):
|
def ping_until_delivered(host, **ping_params):
|
||||||
@ -122,21 +122,21 @@ def ping_until_unreceived(host, **ping_params):
|
|||||||
return ping(host=host, until=UNRECEIVED, **ping_params)
|
return ping(host=host, until=UNRECEIVED, **ping_params)
|
||||||
|
|
||||||
|
|
||||||
def get_statistics(parameters=None, ssh_client=None, until=None,
|
def get_statistics(parameters=None, ssh_client=None, until=None, check=True,
|
||||||
**ping_params):
|
**ping_params):
|
||||||
parameters = _parameters.get_ping_parameters(default=parameters,
|
parameters = _parameters.get_ping_parameters(default=parameters,
|
||||||
**ping_params)
|
**ping_params)
|
||||||
statistics = _statistics.PingStatistics()
|
statistics = _statistics.PingStatistics()
|
||||||
for partial_statistics in iter_statistics(parameters=parameters,
|
for partial_statistics in iter_statistics(parameters=parameters,
|
||||||
ssh_client=ssh_client,
|
ssh_client=ssh_client,
|
||||||
until=until):
|
until=until, check=check):
|
||||||
statistics += partial_statistics
|
statistics += partial_statistics
|
||||||
LOG.debug("%r", statistics)
|
LOG.debug("%r", statistics)
|
||||||
|
|
||||||
return statistics
|
return statistics
|
||||||
|
|
||||||
|
|
||||||
def iter_statistics(parameters=None, ssh_client=None, until=None,
|
def iter_statistics(parameters=None, ssh_client=None, until=None, check=True,
|
||||||
**ping_params):
|
**ping_params):
|
||||||
parameters = _parameters.get_ping_parameters(default=parameters,
|
parameters = _parameters.get_ping_parameters(default=parameters,
|
||||||
**ping_params)
|
**ping_params)
|
||||||
@ -177,7 +177,8 @@ def iter_statistics(parameters=None, ssh_client=None, until=None,
|
|||||||
try:
|
try:
|
||||||
result = execute_ping(parameters=execute_parameters,
|
result = execute_ping(parameters=execute_parameters,
|
||||||
ssh_client=ssh_client,
|
ssh_client=ssh_client,
|
||||||
timeout=end_of_time - now)
|
timeout=end_of_time - now,
|
||||||
|
check=check)
|
||||||
except sh.ShellTimeoutExpired:
|
except sh.ShellTimeoutExpired:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
@ -208,7 +209,7 @@ def iter_statistics(parameters=None, ssh_client=None, until=None,
|
|||||||
message_type=until)
|
message_type=until)
|
||||||
|
|
||||||
|
|
||||||
def execute_ping(parameters, ssh_client=None, **params):
|
def execute_ping(parameters, ssh_client=None, check=True, **params):
|
||||||
if not ssh_client:
|
if not ssh_client:
|
||||||
is_cirros_image = params.setdefault('is_cirros_image', False)
|
is_cirros_image = params.setdefault('is_cirros_image', False)
|
||||||
if is_cirros_image:
|
if is_cirros_image:
|
||||||
@ -218,7 +219,7 @@ def execute_ping(parameters, ssh_client=None, **params):
|
|||||||
command = get_ping_command(parameters)
|
command = get_ping_command(parameters)
|
||||||
result = sh.execute(command=command, ssh_client=ssh_client,
|
result = sh.execute(command=command, ssh_client=ssh_client,
|
||||||
timeout=parameters.timeout, check=False)
|
timeout=parameters.timeout, check=False)
|
||||||
if result.exit_status:
|
if check and result.exit_status:
|
||||||
handle_ping_command_error(error=result.stderr)
|
handle_ping_command_error(error=result.stderr)
|
||||||
return result
|
return result
|
||||||
|
|
||||||
|
@ -16,9 +16,9 @@ from __future__ import absolute_import
|
|||||||
|
|
||||||
import tobiko
|
import tobiko
|
||||||
from tobiko import config
|
from tobiko import config
|
||||||
from tobiko.common.utils import network
|
|
||||||
from tobiko.openstack import heat
|
from tobiko.openstack import heat
|
||||||
from tobiko.tests.scenario.neutron import base
|
from tobiko.tests.scenario.neutron import base
|
||||||
|
from tobiko.shell import ping
|
||||||
|
|
||||||
|
|
||||||
CONF = config.CONF
|
CONF = config.CONF
|
||||||
@ -26,9 +26,12 @@ CONF = config.CONF
|
|||||||
|
|
||||||
class FloatingIPFixture(heat.HeatStackFixture):
|
class FloatingIPFixture(heat.HeatStackFixture):
|
||||||
template = base.heat_template_file('floating_ip.yaml')
|
template = base.heat_template_file('floating_ip.yaml')
|
||||||
|
|
||||||
|
# template parameters
|
||||||
floating_network = CONF.tobiko.neutron.floating_network
|
floating_network = CONF.tobiko.neutron.floating_network
|
||||||
image = CONF.tobiko.nova.image
|
image = CONF.tobiko.nova.image
|
||||||
flavor = CONF.tobiko.nova.flavor
|
flavor = CONF.tobiko.nova.flavor
|
||||||
|
|
||||||
internal_network_fixture = base.InternalNetworkFixture
|
internal_network_fixture = base.InternalNetworkFixture
|
||||||
internal_network = None
|
internal_network = None
|
||||||
|
|
||||||
@ -42,12 +45,66 @@ class FloatingIPFixture(heat.HeatStackFixture):
|
|||||||
self.internal_network_fixture).wait_for_outputs()
|
self.internal_network_fixture).wait_for_outputs()
|
||||||
|
|
||||||
|
|
||||||
|
class FloatingIPTest(base.NeutronTest):
|
||||||
|
"""Tests server connectivity"""
|
||||||
|
|
||||||
|
floating_ip_fixture = FloatingIPFixture
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def setUpClass(cls):
|
||||||
|
super(FloatingIPTest, cls).setUpClass()
|
||||||
|
stack = tobiko.setup_fixture(cls.floating_ip_fixture)
|
||||||
|
outputs = stack.wait_for_outputs()
|
||||||
|
cls.floating_ip_address = outputs.floating_ip_address
|
||||||
|
cls.mtu = stack.internal_network.mtu
|
||||||
|
|
||||||
|
def test_ping(self):
|
||||||
|
ping.ping_until_received(self.floating_ip_address).assert_replied()
|
||||||
|
|
||||||
|
def test_ping_with_mtu_packet(self):
|
||||||
|
ping.ping_until_received(self.floating_ip_address,
|
||||||
|
packet_size=self.mtu,
|
||||||
|
fragmentation=False).assert_replied()
|
||||||
|
|
||||||
|
def test_ping_with_oversized_packet(self):
|
||||||
|
# Wait for VM to get ready
|
||||||
|
ping.ping_until_received(self.floating_ip_address)
|
||||||
|
|
||||||
|
# Send 5 over-sized packets
|
||||||
|
ping.ping(self.floating_ip_address, packet_size=self.mtu + 1,
|
||||||
|
fragmentation=False, count=5,
|
||||||
|
check=False).assert_not_replied()
|
||||||
|
|
||||||
|
|
||||||
class FloatingIPWithPortSecurityFixture(FloatingIPFixture):
|
class FloatingIPWithPortSecurityFixture(FloatingIPFixture):
|
||||||
port_security_enabled = True
|
port_security_enabled = True
|
||||||
|
|
||||||
|
|
||||||
class FloatingIPWithSecurityGroupFixture(FloatingIPFixture):
|
class FloatingIPWithPortSecurityTest(base.NeutronTest):
|
||||||
port_security_enabled = True
|
floating_ip_fixture = FloatingIPFixture
|
||||||
|
floating_ip_with_securtity_fixture = FloatingIPWithPortSecurityFixture
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def setUpClass(cls):
|
||||||
|
super(FloatingIPWithPortSecurityTest, cls).setUpClass()
|
||||||
|
|
||||||
|
# Setup VM with port security
|
||||||
|
stack = tobiko.setup_fixture(cls.floating_ip_with_securtity_fixture)
|
||||||
|
outputs = stack.wait_for_outputs()
|
||||||
|
cls.floating_ip_address_with_security = outputs.floating_ip_address
|
||||||
|
|
||||||
|
# Setup VM without port security
|
||||||
|
stack = tobiko.setup_fixture(cls.floating_ip_fixture)
|
||||||
|
outputs = stack.wait_for_outputs()
|
||||||
|
cls.floating_ip_address = outputs.floating_ip_address
|
||||||
|
|
||||||
|
def test_ping(self):
|
||||||
|
ping.ping_until_received(self.floating_ip_address).assert_replied()
|
||||||
|
ping.ping(self.floating_ip_address_with_security,
|
||||||
|
count=5).assert_not_replied()
|
||||||
|
|
||||||
|
|
||||||
|
class FloatingIPWithSecurityGroupFixture(FloatingIPWithPortSecurityFixture):
|
||||||
security_groups_fixture = base.SecurityGroupsFixture
|
security_groups_fixture = base.SecurityGroupsFixture
|
||||||
security_groups = None
|
security_groups = None
|
||||||
|
|
||||||
@ -62,29 +119,5 @@ class FloatingIPWithSecurityGroupFixture(FloatingIPFixture):
|
|||||||
self.security_groups_fixture).wait_for_outputs()
|
self.security_groups_fixture).wait_for_outputs()
|
||||||
|
|
||||||
|
|
||||||
class FloatingIPTest(base.NeutronTest):
|
class FloatingIPWithSecurityGroupTest(FloatingIPTest):
|
||||||
"""Tests server connectivity"""
|
floating_ip_fixture = FloatingIPWithSecurityGroupFixture
|
||||||
|
|
||||||
def test_ping_floating_ip(self, fixture_type=FloatingIPFixture):
|
|
||||||
"""Validates connectivity to a server post upgrade."""
|
|
||||||
stack = self.setup_fixture(fixture_type)
|
|
||||||
network.assert_ping(stack.outputs.floating_ip_address)
|
|
||||||
|
|
||||||
def test_ping_floating_ip_with_port_security(
|
|
||||||
self, fixture_type=FloatingIPWithPortSecurityFixture):
|
|
||||||
"""Validates connectivity to a server post upgrade."""
|
|
||||||
stack = self.setup_fixture(fixture_type)
|
|
||||||
network.assert_ping(stack.outputs.floating_ip_address,
|
|
||||||
should_fail=True)
|
|
||||||
|
|
||||||
def test_ping_floating_ip_with_security_group(
|
|
||||||
self, fixture_type=FloatingIPWithSecurityGroupFixture):
|
|
||||||
"""Validates connectivity to a server post upgrade."""
|
|
||||||
stack = self.setup_fixture(fixture_type)
|
|
||||||
network.assert_ping(stack.outputs.floating_ip_address)
|
|
||||||
|
|
||||||
def test_ping_with_oversize_packet(self, fixture_type=FloatingIPFixture):
|
|
||||||
stack = self.setup_fixture(fixture_type)
|
|
||||||
network.assert_ping(stack.outputs.floating_ip_address,
|
|
||||||
packet_size=stack.internal_network.mtu + 1,
|
|
||||||
fragmentation=False, should_fail=True)
|
|
||||||
|
Loading…
Reference in New Issue
Block a user