ZhiQiang Fan | 39f9722 | 2013-09-20 04:49:44 +0800 | [diff] [blame] | 1 | # Copyright 2012 OpenStack Foundation |
Gavin Brebner | 0f465a3 | 2013-03-14 13:26:09 +0000 | [diff] [blame] | 2 | # Copyright 2013 Hewlett-Packard Development Company, L.P. |
Maru Newby | 81f07a0 | 2012-09-05 20:21:19 -0700 | [diff] [blame] | 3 | # All Rights Reserved. |
| 4 | # |
| 5 | # Licensed under the Apache License, Version 2.0 (the "License"); you may |
| 6 | # not use this file except in compliance with the License. You may obtain |
| 7 | # a copy of the License at |
| 8 | # |
| 9 | # http://www.apache.org/licenses/LICENSE-2.0 |
| 10 | # |
| 11 | # Unless required by applicable law or agreed to in writing, software |
| 12 | # distributed under the License is distributed on an "AS IS" BASIS, WITHOUT |
| 13 | # WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the |
| 14 | # License for the specific language governing permissions and limitations |
| 15 | # under the License. |
| 16 | |
Attila Fazekas | aeeeefd | 2013-08-06 17:01:56 +0200 | [diff] [blame] | 17 | from tempest.common import debug |
Masayuki Igawa | 259c113 | 2013-10-31 17:48:44 +0900 | [diff] [blame] | 18 | from tempest.common.utils import data_utils |
Matthew Treinish | cb56994 | 2013-08-09 16:33:44 -0400 | [diff] [blame] | 19 | from tempest import config |
Matthew Treinish | 2b59f84 | 2013-09-09 20:32:51 +0000 | [diff] [blame] | 20 | from tempest.openstack.common import log as logging |
Sean Dague | 6dbc6da | 2013-05-08 17:49:46 -0400 | [diff] [blame] | 21 | from tempest.scenario import manager |
Brent Eagles | c26d452 | 2013-12-02 13:28:49 -0500 | [diff] [blame] | 22 | |
Maru Newby | bd36022 | 2013-04-08 22:48:50 +0000 | [diff] [blame] | 23 | from tempest.test import attr |
Matthew Treinish | 2153ec0 | 2013-09-09 20:57:30 +0000 | [diff] [blame] | 24 | from tempest.test import services |
Maru Newby | 81f07a0 | 2012-09-05 20:21:19 -0700 | [diff] [blame] | 25 | |
Sean Dague | 86bd842 | 2013-12-20 09:56:44 -0500 | [diff] [blame] | 26 | CONF = config.CONF |
Matthew Treinish | 2b59f84 | 2013-09-09 20:32:51 +0000 | [diff] [blame] | 27 | LOG = logging.getLogger(__name__) |
| 28 | |
Maru Newby | 81f07a0 | 2012-09-05 20:21:19 -0700 | [diff] [blame] | 29 | |
Sean Dague | 6dbc6da | 2013-05-08 17:49:46 -0400 | [diff] [blame] | 30 | class TestNetworkBasicOps(manager.NetworkScenarioTest): |
Maru Newby | 81f07a0 | 2012-09-05 20:21:19 -0700 | [diff] [blame] | 31 | |
| 32 | """ |
| 33 | This smoke test suite assumes that Nova has been configured to |
Mark McClain | f2982e8 | 2013-07-06 17:48:03 -0400 | [diff] [blame] | 34 | boot VM's with Neutron-managed networking, and attempts to |
Maru Newby | 81f07a0 | 2012-09-05 20:21:19 -0700 | [diff] [blame] | 35 | verify network connectivity as follows: |
| 36 | |
| 37 | * For a freshly-booted VM with an IP address ("port") on a given network: |
| 38 | |
Maru Newby | af292e8 | 2013-05-20 21:32:28 +0000 | [diff] [blame] | 39 | - the Tempest host can ping the IP address. This implies, but |
| 40 | does not guarantee (see the ssh check that follows), that the |
| 41 | VM has been assigned the correct IP address and has |
Maru Newby | 81f07a0 | 2012-09-05 20:21:19 -0700 | [diff] [blame] | 42 | connectivity to the Tempest host. |
| 43 | |
Maru Newby | af292e8 | 2013-05-20 21:32:28 +0000 | [diff] [blame] | 44 | - the Tempest host can perform key-based authentication to an |
| 45 | ssh server hosted at the IP address. This check guarantees |
| 46 | that the IP address is associated with the target VM. |
| 47 | |
Yair Fried | 9a551c4 | 2013-12-15 14:59:34 +0200 | [diff] [blame] | 48 | - detach the floating-ip from the VM and verify that it becomes |
| 49 | unreachable |
| 50 | |
Yair Fried | 05db252 | 2013-11-18 11:02:10 +0200 | [diff] [blame] | 51 | - associate detached floating ip to a new VM and verify connectivity. |
| 52 | VMs are created with unique keypair so connectivity also asserts that |
| 53 | floating IP is associated with the new VM instead of the old one |
| 54 | |
Attila Fazekas | c3a095b | 2013-08-17 09:15:44 +0200 | [diff] [blame] | 55 | # TODO(mnewby) - Need to implement the following: |
Maru Newby | 81f07a0 | 2012-09-05 20:21:19 -0700 | [diff] [blame] | 56 | - the Tempest host can ssh into the VM via the IP address and |
| 57 | successfully execute the following: |
| 58 | |
| 59 | - ping an external IP address, implying external connectivity. |
| 60 | |
| 61 | - ping an external hostname, implying that dns is correctly |
| 62 | configured. |
| 63 | |
| 64 | - ping an internal IP address, implying connectivity to another |
| 65 | VM on the same network. |
| 66 | |
| 67 | There are presumed to be two types of networks: tenant and |
| 68 | public. A tenant network may or may not be reachable from the |
| 69 | Tempest host. A public network is assumed to be reachable from |
| 70 | the Tempest host, and it should be possible to associate a public |
| 71 | ('floating') IP address with a tenant ('fixed') IP address to |
Chang Bo Guo | cc1623c | 2013-09-13 20:11:27 -0700 | [diff] [blame] | 72 | facilitate external connectivity to a potentially unroutable |
Maru Newby | 81f07a0 | 2012-09-05 20:21:19 -0700 | [diff] [blame] | 73 | tenant IP address. |
| 74 | |
| 75 | This test suite can be configured to test network connectivity to |
| 76 | a VM via a tenant network, a public network, or both. If both |
| 77 | networking types are to be evaluated, tests that need to be |
| 78 | executed remotely on the VM (via ssh) will only be run against |
| 79 | one of the networks (to minimize test execution time). |
| 80 | |
| 81 | Determine which types of networks to test as follows: |
| 82 | |
| 83 | * Configure tenant network checks (via the |
| 84 | 'tenant_networks_reachable' key) if the Tempest host should |
| 85 | have direct connectivity to tenant networks. This is likely to |
| 86 | be the case if Tempest is running on the same host as a |
| 87 | single-node devstack installation with IP namespaces disabled. |
| 88 | |
| 89 | * Configure checks for a public network if a public network has |
| 90 | been configured prior to the test suite being run and if the |
| 91 | Tempest host should have connectivity to that public network. |
| 92 | Checking connectivity for a public network requires that a |
| 93 | value be provided for 'public_network_id'. A value can |
| 94 | optionally be provided for 'public_router_id' if tenants will |
| 95 | use a shared router to access a public network (as is likely to |
| 96 | be the case when IP namespaces are not enabled). If a value is |
| 97 | not provided for 'public_router_id', a router will be created |
| 98 | for each tenant and use the network identified by |
| 99 | 'public_network_id' as its gateway. |
| 100 | |
| 101 | """ |
| 102 | |
| 103 | @classmethod |
| 104 | def check_preconditions(cls): |
Gavin Brebner | 0f465a3 | 2013-03-14 13:26:09 +0000 | [diff] [blame] | 105 | super(TestNetworkBasicOps, cls).check_preconditions() |
Matthew Treinish | 6c07229 | 2014-01-29 19:15:52 +0000 | [diff] [blame] | 106 | if not (CONF.network.tenant_networks_reachable |
| 107 | or CONF.network.public_network_id): |
Maru Newby | 81f07a0 | 2012-09-05 20:21:19 -0700 | [diff] [blame] | 108 | msg = ('Either tenant_networks_reachable must be "true", or ' |
| 109 | 'public_network_id must be defined.') |
Gavin Brebner | 0f465a3 | 2013-03-14 13:26:09 +0000 | [diff] [blame] | 110 | cls.enabled = False |
ivan-zhu | 1feeb38 | 2013-01-24 10:14:39 +0800 | [diff] [blame] | 111 | raise cls.skipException(msg) |
Maru Newby | 81f07a0 | 2012-09-05 20:21:19 -0700 | [diff] [blame] | 112 | |
| 113 | @classmethod |
| 114 | def setUpClass(cls): |
| 115 | super(TestNetworkBasicOps, cls).setUpClass() |
| 116 | cls.check_preconditions() |
Maru Newby | 81f07a0 | 2012-09-05 20:21:19 -0700 | [diff] [blame] | 117 | # TODO(mnewby) Consider looking up entities as needed instead |
| 118 | # of storing them as collections on the class. |
Maru Newby | 81f07a0 | 2012-09-05 20:21:19 -0700 | [diff] [blame] | 119 | cls.security_groups = {} |
| 120 | cls.networks = [] |
Gavin Brebner | 851c350 | 2013-01-18 13:14:10 +0000 | [diff] [blame] | 121 | cls.subnets = [] |
| 122 | cls.routers = [] |
Yair Fried | 05db252 | 2013-11-18 11:02:10 +0200 | [diff] [blame] | 123 | cls.servers = {} |
Maru Newby | 81f07a0 | 2012-09-05 20:21:19 -0700 | [diff] [blame] | 124 | cls.floating_ips = {} |
| 125 | |
Matthew Treinish | 2b59f84 | 2013-09-09 20:32:51 +0000 | [diff] [blame] | 126 | def _create_security_groups(self): |
Yair Fried | eb69f3f | 2013-10-10 13:18:16 +0300 | [diff] [blame] | 127 | self.security_groups[self.tenant_id] =\ |
| 128 | self._create_security_group_neutron(tenant_id=self.tenant_id) |
Maru Newby | 81f07a0 | 2012-09-05 20:21:19 -0700 | [diff] [blame] | 129 | |
Matthew Treinish | 2b59f84 | 2013-09-09 20:32:51 +0000 | [diff] [blame] | 130 | def _check_networks(self): |
Attila Fazekas | c3a095b | 2013-08-17 09:15:44 +0200 | [diff] [blame] | 131 | # Checks that we see the newly created network/subnet/router via |
| 132 | # checking the result of list_[networks,routers,subnets] |
Gavin Brebner | 851c350 | 2013-01-18 13:14:10 +0000 | [diff] [blame] | 133 | seen_nets = self._list_networks() |
| 134 | seen_names = [n['name'] for n in seen_nets] |
| 135 | seen_ids = [n['id'] for n in seen_nets] |
| 136 | for mynet in self.networks: |
Gavin Brebner | 0f465a3 | 2013-03-14 13:26:09 +0000 | [diff] [blame] | 137 | self.assertIn(mynet.name, seen_names) |
| 138 | self.assertIn(mynet.id, seen_ids) |
Gavin Brebner | 851c350 | 2013-01-18 13:14:10 +0000 | [diff] [blame] | 139 | seen_subnets = self._list_subnets() |
| 140 | seen_net_ids = [n['network_id'] for n in seen_subnets] |
| 141 | seen_subnet_ids = [n['id'] for n in seen_subnets] |
| 142 | for mynet in self.networks: |
Gavin Brebner | 0f465a3 | 2013-03-14 13:26:09 +0000 | [diff] [blame] | 143 | self.assertIn(mynet.id, seen_net_ids) |
Gavin Brebner | 851c350 | 2013-01-18 13:14:10 +0000 | [diff] [blame] | 144 | for mysubnet in self.subnets: |
Gavin Brebner | 0f465a3 | 2013-03-14 13:26:09 +0000 | [diff] [blame] | 145 | self.assertIn(mysubnet.id, seen_subnet_ids) |
Gavin Brebner | 851c350 | 2013-01-18 13:14:10 +0000 | [diff] [blame] | 146 | seen_routers = self._list_routers() |
| 147 | seen_router_ids = [n['id'] for n in seen_routers] |
| 148 | seen_router_names = [n['name'] for n in seen_routers] |
| 149 | for myrouter in self.routers: |
Gavin Brebner | 0f465a3 | 2013-03-14 13:26:09 +0000 | [diff] [blame] | 150 | self.assertIn(myrouter.name, seen_router_names) |
| 151 | self.assertIn(myrouter.id, seen_router_ids) |
Gavin Brebner | 851c350 | 2013-01-18 13:14:10 +0000 | [diff] [blame] | 152 | |
Salvatore Orlando | 5ed3b6e | 2013-09-17 01:27:26 -0700 | [diff] [blame] | 153 | def _create_server(self, name, network): |
| 154 | tenant_id = network.tenant_id |
Yair Fried | 05db252 | 2013-11-18 11:02:10 +0200 | [diff] [blame] | 155 | keypair = self.create_keypair(name='keypair-%s' % name) |
Salvatore Orlando | 5ed3b6e | 2013-09-17 01:27:26 -0700 | [diff] [blame] | 156 | security_groups = [self.security_groups[tenant_id].name] |
| 157 | create_kwargs = { |
| 158 | 'nics': [ |
| 159 | {'net-id': network.id}, |
| 160 | ], |
Yair Fried | 05db252 | 2013-11-18 11:02:10 +0200 | [diff] [blame] | 161 | 'key_name': keypair.name, |
Salvatore Orlando | 5ed3b6e | 2013-09-17 01:27:26 -0700 | [diff] [blame] | 162 | 'security_groups': security_groups, |
| 163 | } |
Giulio Fidente | 61cadca | 2013-09-24 18:33:37 +0200 | [diff] [blame] | 164 | server = self.create_server(name=name, create_kwargs=create_kwargs) |
Yair Fried | 05db252 | 2013-11-18 11:02:10 +0200 | [diff] [blame] | 165 | self.servers[server] = keypair |
Salvatore Orlando | 5ed3b6e | 2013-09-17 01:27:26 -0700 | [diff] [blame] | 166 | return server |
| 167 | |
Matthew Treinish | 2b59f84 | 2013-09-09 20:32:51 +0000 | [diff] [blame] | 168 | def _create_servers(self): |
Maru Newby | 81f07a0 | 2012-09-05 20:21:19 -0700 | [diff] [blame] | 169 | for i, network in enumerate(self.networks): |
Masayuki Igawa | 259c113 | 2013-10-31 17:48:44 +0900 | [diff] [blame] | 170 | name = data_utils.rand_name('server-smoke-%d-' % i) |
Yair Fried | 05db252 | 2013-11-18 11:02:10 +0200 | [diff] [blame] | 171 | self._create_server(name, network) |
Maru Newby | 81f07a0 | 2012-09-05 20:21:19 -0700 | [diff] [blame] | 172 | |
Matthew Treinish | 2b59f84 | 2013-09-09 20:32:51 +0000 | [diff] [blame] | 173 | def _check_tenant_network_connectivity(self): |
Sean Dague | 86bd842 | 2013-12-20 09:56:44 -0500 | [diff] [blame] | 174 | if not CONF.network.tenant_networks_reachable: |
Maru Newby | 81f07a0 | 2012-09-05 20:21:19 -0700 | [diff] [blame] | 175 | msg = 'Tenant networks not configured to be reachable.' |
Matthew Treinish | 2b59f84 | 2013-09-09 20:32:51 +0000 | [diff] [blame] | 176 | LOG.info(msg) |
| 177 | return |
Maru Newby | af292e8 | 2013-05-20 21:32:28 +0000 | [diff] [blame] | 178 | # The target login is assumed to have been configured for |
| 179 | # key-based authentication by cloud-init. |
Sean Dague | 86bd842 | 2013-12-20 09:56:44 -0500 | [diff] [blame] | 180 | ssh_login = CONF.compute.image_ssh_user |
Brent Eagles | c26d452 | 2013-12-02 13:28:49 -0500 | [diff] [blame] | 181 | try: |
Yair Fried | 05db252 | 2013-11-18 11:02:10 +0200 | [diff] [blame] | 182 | for server, key in self.servers.items(): |
Brent Eagles | c26d452 | 2013-12-02 13:28:49 -0500 | [diff] [blame] | 183 | for net_name, ip_addresses in server.networks.iteritems(): |
| 184 | for ip_address in ip_addresses: |
| 185 | self._check_vm_connectivity(ip_address, ssh_login, |
Yair Fried | 05db252 | 2013-11-18 11:02:10 +0200 | [diff] [blame] | 186 | key.private_key) |
Yair Fried | a039f87 | 2014-01-02 12:11:10 +0200 | [diff] [blame] | 187 | except Exception: |
| 188 | LOG.exception('Tenant connectivity check failed') |
Salvatore Orlando | 728d916 | 2014-01-19 10:33:23 -0800 | [diff] [blame] | 189 | self._log_console_output(servers=self.servers.keys()) |
Brent Eagles | c26d452 | 2013-12-02 13:28:49 -0500 | [diff] [blame] | 190 | debug.log_ip_ns() |
Yair Fried | a039f87 | 2014-01-02 12:11:10 +0200 | [diff] [blame] | 191 | raise |
Brent Eagles | c26d452 | 2013-12-02 13:28:49 -0500 | [diff] [blame] | 192 | |
Yair Fried | 9a551c4 | 2013-12-15 14:59:34 +0200 | [diff] [blame] | 193 | def _create_and_associate_floating_ips(self): |
Sean Dague | 86bd842 | 2013-12-20 09:56:44 -0500 | [diff] [blame] | 194 | public_network_id = CONF.network.public_network_id |
Yair Fried | 05db252 | 2013-11-18 11:02:10 +0200 | [diff] [blame] | 195 | for server in self.servers.keys(): |
Maru Newby | 81f07a0 | 2012-09-05 20:21:19 -0700 | [diff] [blame] | 196 | floating_ip = self._create_floating_ip(server, public_network_id) |
Yair Fried | 9a551c4 | 2013-12-15 14:59:34 +0200 | [diff] [blame] | 197 | self.floating_ips[floating_ip] = server |
Maru Newby | 81f07a0 | 2012-09-05 20:21:19 -0700 | [diff] [blame] | 198 | |
Yair Fried | a4b0d1d | 2014-01-26 13:59:54 +0200 | [diff] [blame] | 199 | def _check_public_network_connectivity(self, should_connect=True, |
| 200 | msg=None): |
Maru Newby | af292e8 | 2013-05-20 21:32:28 +0000 | [diff] [blame] | 201 | # The target login is assumed to have been configured for |
| 202 | # key-based authentication by cloud-init. |
Sean Dague | 86bd842 | 2013-12-20 09:56:44 -0500 | [diff] [blame] | 203 | ssh_login = CONF.compute.image_ssh_user |
Nachi Ueno | 4fb7ce6 | 2014-01-09 18:29:34 -0800 | [diff] [blame] | 204 | LOG.debug('checking network connections') |
Attila Fazekas | aeeeefd | 2013-08-06 17:01:56 +0200 | [diff] [blame] | 205 | try: |
Yair Fried | 9a551c4 | 2013-12-15 14:59:34 +0200 | [diff] [blame] | 206 | for floating_ip, server in self.floating_ips.iteritems(): |
| 207 | ip_address = floating_ip.floating_ip_address |
Yair Fried | 05db252 | 2013-11-18 11:02:10 +0200 | [diff] [blame] | 208 | private_key = None |
| 209 | if should_connect: |
| 210 | private_key = self.servers[server].private_key |
Yair Fried | 9a551c4 | 2013-12-15 14:59:34 +0200 | [diff] [blame] | 211 | self._check_vm_connectivity(ip_address, |
| 212 | ssh_login, |
| 213 | private_key, |
| 214 | should_connect=should_connect) |
Yair Fried | a039f87 | 2014-01-02 12:11:10 +0200 | [diff] [blame] | 215 | except Exception: |
Yair Fried | a4b0d1d | 2014-01-26 13:59:54 +0200 | [diff] [blame] | 216 | ex_msg = 'Public network connectivity check failed' |
| 217 | if msg: |
| 218 | ex_msg += ": " + msg |
| 219 | LOG.exception(ex_msg) |
Salvatore Orlando | 728d916 | 2014-01-19 10:33:23 -0800 | [diff] [blame] | 220 | self._log_console_output(servers=self.servers.keys()) |
Attila Fazekas | aeeeefd | 2013-08-06 17:01:56 +0200 | [diff] [blame] | 221 | debug.log_ip_ns() |
Yair Fried | a039f87 | 2014-01-02 12:11:10 +0200 | [diff] [blame] | 222 | raise |
Matthew Treinish | 2b59f84 | 2013-09-09 20:32:51 +0000 | [diff] [blame] | 223 | |
Yair Fried | 9a551c4 | 2013-12-15 14:59:34 +0200 | [diff] [blame] | 224 | def _disassociate_floating_ips(self): |
| 225 | for floating_ip, server in self.floating_ips.iteritems(): |
| 226 | self._disassociate_floating_ip(floating_ip) |
| 227 | self.floating_ips[floating_ip] = None |
| 228 | |
Yair Fried | 05db252 | 2013-11-18 11:02:10 +0200 | [diff] [blame] | 229 | def _reassociate_floating_ips(self): |
| 230 | network = self.networks[0] |
| 231 | for floating_ip in self.floating_ips.keys(): |
| 232 | name = data_utils.rand_name('new_server-smoke-') |
| 233 | # create a new server for the floating ip |
| 234 | server = self._create_server(name, network) |
| 235 | self._associate_floating_ip(floating_ip, server) |
| 236 | self.floating_ips[floating_ip] = server |
| 237 | |
Matthew Treinish | 2b59f84 | 2013-09-09 20:32:51 +0000 | [diff] [blame] | 238 | @attr(type='smoke') |
Matthew Treinish | 2153ec0 | 2013-09-09 20:57:30 +0000 | [diff] [blame] | 239 | @services('compute', 'network') |
Matthew Treinish | 2b59f84 | 2013-09-09 20:32:51 +0000 | [diff] [blame] | 240 | def test_network_basic_ops(self): |
Matthew Treinish | 2b59f84 | 2013-09-09 20:32:51 +0000 | [diff] [blame] | 241 | self._create_security_groups() |
| 242 | self._create_networks() |
| 243 | self._check_networks() |
| 244 | self._create_servers() |
Yair Fried | 9a551c4 | 2013-12-15 14:59:34 +0200 | [diff] [blame] | 245 | self._create_and_associate_floating_ips() |
Salvatore Orlando | 8d076cb | 2013-09-16 11:00:44 -0700 | [diff] [blame] | 246 | self._check_tenant_network_connectivity() |
Yair Fried | 9a551c4 | 2013-12-15 14:59:34 +0200 | [diff] [blame] | 247 | self._check_public_network_connectivity(should_connect=True) |
| 248 | self._disassociate_floating_ips() |
Yair Fried | a4b0d1d | 2014-01-26 13:59:54 +0200 | [diff] [blame] | 249 | self._check_public_network_connectivity(should_connect=False, |
| 250 | msg="after disassociate " |
| 251 | "floating ip") |
Yair Fried | 05db252 | 2013-11-18 11:02:10 +0200 | [diff] [blame] | 252 | self._reassociate_floating_ips() |
Yair Fried | a4b0d1d | 2014-01-26 13:59:54 +0200 | [diff] [blame] | 253 | self._check_public_network_connectivity(should_connect=True, |
| 254 | msg="after re-associate " |
| 255 | "floating ip") |