ZhiQiang Fan | 39f9722 | 2013-09-20 04:49:44 +0800 | [diff] [blame] | 1 | # Copyright 2012 OpenStack Foundation |
Jay Pipes | 051075a | 2012-04-28 17:39:37 -0400 | [diff] [blame] | 2 | # All Rights Reserved. |
| 3 | # |
| 4 | # Licensed under the Apache License, Version 2.0 (the "License"); you may |
| 5 | # not use this file except in compliance with the License. You may obtain |
| 6 | # a copy of the License at |
| 7 | # |
| 8 | # http://www.apache.org/licenses/LICENSE-2.0 |
| 9 | # |
| 10 | # Unless required by applicable law or agreed to in writing, software |
| 11 | # distributed under the License is distributed on an "AS IS" BASIS, WITHOUT |
| 12 | # WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the |
| 13 | # License for the specific language governing permissions and limitations |
| 14 | # under the License. |
| 15 | |
Masayuki Igawa | 259c113 | 2013-10-31 17:48:44 +0900 | [diff] [blame] | 16 | from tempest.common.utils import data_utils |
Andrea Frittoli | f5da28b | 2013-12-06 07:08:07 +0000 | [diff] [blame] | 17 | from tempest.common.utils import test_utils |
Matthew Treinish | 6c07229 | 2014-01-29 19:15:52 +0000 | [diff] [blame] | 18 | from tempest import config |
Matthew Treinish | f4a9b0f | 2013-07-26 16:58:26 -0400 | [diff] [blame] | 19 | from tempest.openstack.common import log as logging |
Sean Dague | 6dbc6da | 2013-05-08 17:49:46 -0400 | [diff] [blame] | 20 | from tempest.scenario import manager |
Masayuki Igawa | 4ded9f0 | 2014-02-17 15:05:59 +0900 | [diff] [blame] | 21 | from tempest import test |
Jay Pipes | 051075a | 2012-04-28 17:39:37 -0400 | [diff] [blame] | 22 | |
Andrea Frittoli | f5da28b | 2013-12-06 07:08:07 +0000 | [diff] [blame] | 23 | import testscenarios |
| 24 | |
Matthew Treinish | 6c07229 | 2014-01-29 19:15:52 +0000 | [diff] [blame] | 25 | CONF = config.CONF |
| 26 | |
Jay Pipes | 051075a | 2012-04-28 17:39:37 -0400 | [diff] [blame] | 27 | LOG = logging.getLogger(__name__) |
| 28 | |
Andrea Frittoli | f5da28b | 2013-12-06 07:08:07 +0000 | [diff] [blame] | 29 | load_tests = testscenarios.load_tests_apply_scenarios |
| 30 | |
Jay Pipes | 051075a | 2012-04-28 17:39:37 -0400 | [diff] [blame] | 31 | |
Sean Dague | 6dbc6da | 2013-05-08 17:49:46 -0400 | [diff] [blame] | 32 | class TestServerBasicOps(manager.OfficialClientTest): |
Jay Pipes | 051075a | 2012-04-28 17:39:37 -0400 | [diff] [blame] | 33 | |
| 34 | """ |
| 35 | This smoke test case follows this basic set of operations: |
| 36 | |
| 37 | * Create a keypair for use in launching an instance |
| 38 | * Create a security group to control network access in instance |
| 39 | * Add simple permissive rules to the security group |
| 40 | * Launch an instance |
| 41 | * Pause/unpause the instance |
| 42 | * Suspend/resume the instance |
| 43 | * Terminate the instance |
| 44 | """ |
| 45 | |
Andrea Frittoli | f5da28b | 2013-12-06 07:08:07 +0000 | [diff] [blame] | 46 | scenario_utils = test_utils.InputScenarioUtils() |
| 47 | scenario_flavor = scenario_utils.scenario_flavors |
| 48 | scenario_image = scenario_utils.scenario_images |
| 49 | |
| 50 | scenarios = testscenarios.multiply_scenarios(scenario_image, |
| 51 | scenario_flavor) |
| 52 | |
| 53 | def setUp(self): |
| 54 | super(TestServerBasicOps, self).setUp() |
| 55 | # Setup image and flavor the test instance |
| 56 | # Support both configured and injected values |
| 57 | if not hasattr(self, 'image_ref'): |
Matthew Treinish | 6c07229 | 2014-01-29 19:15:52 +0000 | [diff] [blame] | 58 | self.image_ref = CONF.compute.image_ref |
Andrea Frittoli | f5da28b | 2013-12-06 07:08:07 +0000 | [diff] [blame] | 59 | if not hasattr(self, 'flavor_ref'): |
Matthew Treinish | 6c07229 | 2014-01-29 19:15:52 +0000 | [diff] [blame] | 60 | self.flavor_ref = CONF.compute.flavor_ref |
Andrea Frittoli | f5da28b | 2013-12-06 07:08:07 +0000 | [diff] [blame] | 61 | self.image_utils = test_utils.ImageUtils() |
| 62 | if not self.image_utils.is_flavor_enough(self.flavor_ref, |
| 63 | self.image_ref): |
| 64 | raise self.skipException( |
| 65 | '{image} does not fit in {flavor}'.format( |
| 66 | image=self.image_ref, flavor=self.flavor_ref |
| 67 | ) |
| 68 | ) |
Matthew Treinish | 6c07229 | 2014-01-29 19:15:52 +0000 | [diff] [blame] | 69 | self.run_ssh = CONF.compute.run_ssh and \ |
Andrea Frittoli | f5da28b | 2013-12-06 07:08:07 +0000 | [diff] [blame] | 70 | self.image_utils.is_sshable_image(self.image_ref) |
| 71 | self.ssh_user = self.image_utils.ssh_user(self.image_ref) |
| 72 | LOG.debug('Starting test for i:{image}, f:{flavor}. ' |
| 73 | 'Run ssh: {ssh}, user: {ssh_user}'.format( |
| 74 | image=self.image_ref, flavor=self.flavor_ref, |
| 75 | ssh=self.run_ssh, ssh_user=self.ssh_user)) |
| 76 | |
Ken'ichi Ohmichi | 599d1b8 | 2013-08-19 18:48:37 +0900 | [diff] [blame] | 77 | def add_keypair(self): |
| 78 | self.keypair = self.create_keypair() |
Jay Pipes | 051075a | 2012-04-28 17:39:37 -0400 | [diff] [blame] | 79 | |
ivan-zhu | 1997739 | 2013-01-12 21:57:55 +0800 | [diff] [blame] | 80 | def create_security_group(self): |
Masayuki Igawa | 259c113 | 2013-10-31 17:48:44 +0900 | [diff] [blame] | 81 | sg_name = data_utils.rand_name('secgroup-smoke') |
Jay Pipes | 051075a | 2012-04-28 17:39:37 -0400 | [diff] [blame] | 82 | sg_desc = sg_name + " description" |
Maru Newby | dec13ec | 2012-08-30 11:19:17 -0700 | [diff] [blame] | 83 | self.secgroup = self.compute_client.security_groups.create(sg_name, |
| 84 | sg_desc) |
Giulio Fidente | 92f7719 | 2013-08-26 17:13:28 +0200 | [diff] [blame] | 85 | self.assertEqual(self.secgroup.name, sg_name) |
| 86 | self.assertEqual(self.secgroup.description, sg_desc) |
| 87 | self.set_resource('secgroup', self.secgroup) |
Jay Pipes | 051075a | 2012-04-28 17:39:37 -0400 | [diff] [blame] | 88 | |
| 89 | # Add rules to the security group |
Yair Fried | eb69f3f | 2013-10-10 13:18:16 +0300 | [diff] [blame] | 90 | self._create_loginable_secgroup_rule_nova(secgroup_id=self.secgroup.id) |
Jay Pipes | 051075a | 2012-04-28 17:39:37 -0400 | [diff] [blame] | 91 | |
ivan-zhu | 1997739 | 2013-01-12 21:57:55 +0800 | [diff] [blame] | 92 | def boot_instance(self): |
Andrea Frittoli | f5da28b | 2013-12-06 07:08:07 +0000 | [diff] [blame] | 93 | # Create server with image and flavor from input scenario |
Jay Pipes | 051075a | 2012-04-28 17:39:37 -0400 | [diff] [blame] | 94 | create_kwargs = { |
Ken'ichi Ohmichi | 599d1b8 | 2013-08-19 18:48:37 +0900 | [diff] [blame] | 95 | 'key_name': self.keypair.id |
Jay Pipes | 051075a | 2012-04-28 17:39:37 -0400 | [diff] [blame] | 96 | } |
Andrea Frittoli | f5da28b | 2013-12-06 07:08:07 +0000 | [diff] [blame] | 97 | instance = self.create_server(image=self.image_ref, |
| 98 | flavor=self.flavor_ref, |
| 99 | create_kwargs=create_kwargs) |
Ken'ichi Ohmichi | 61f272b | 2013-08-15 15:58:53 +0900 | [diff] [blame] | 100 | self.set_resource('instance', instance) |
Jay Pipes | 051075a | 2012-04-28 17:39:37 -0400 | [diff] [blame] | 101 | |
ivan-zhu | 1997739 | 2013-01-12 21:57:55 +0800 | [diff] [blame] | 102 | def pause_server(self): |
Jay Pipes | 051075a | 2012-04-28 17:39:37 -0400 | [diff] [blame] | 103 | instance = self.get_resource('instance') |
| 104 | instance_id = instance.id |
| 105 | LOG.debug("Pausing instance %s. Current status: %s", |
| 106 | instance_id, instance.status) |
| 107 | instance.pause() |
Sean Dague | 35a7caf | 2013-05-10 10:38:22 -0400 | [diff] [blame] | 108 | self.status_timeout( |
| 109 | self.compute_client.servers, instance_id, 'PAUSED') |
Jay Pipes | 051075a | 2012-04-28 17:39:37 -0400 | [diff] [blame] | 110 | |
ivan-zhu | 1997739 | 2013-01-12 21:57:55 +0800 | [diff] [blame] | 111 | def unpause_server(self): |
Jay Pipes | 051075a | 2012-04-28 17:39:37 -0400 | [diff] [blame] | 112 | instance = self.get_resource('instance') |
| 113 | instance_id = instance.id |
| 114 | LOG.debug("Unpausing instance %s. Current status: %s", |
| 115 | instance_id, instance.status) |
| 116 | instance.unpause() |
Sean Dague | 35a7caf | 2013-05-10 10:38:22 -0400 | [diff] [blame] | 117 | self.status_timeout( |
| 118 | self.compute_client.servers, instance_id, 'ACTIVE') |
Jay Pipes | 051075a | 2012-04-28 17:39:37 -0400 | [diff] [blame] | 119 | |
ivan-zhu | 1997739 | 2013-01-12 21:57:55 +0800 | [diff] [blame] | 120 | def suspend_server(self): |
Jay Pipes | 051075a | 2012-04-28 17:39:37 -0400 | [diff] [blame] | 121 | instance = self.get_resource('instance') |
| 122 | instance_id = instance.id |
| 123 | LOG.debug("Suspending instance %s. Current status: %s", |
| 124 | instance_id, instance.status) |
| 125 | instance.suspend() |
Sean Dague | 35a7caf | 2013-05-10 10:38:22 -0400 | [diff] [blame] | 126 | self.status_timeout(self.compute_client.servers, |
Maru Newby | dec13ec | 2012-08-30 11:19:17 -0700 | [diff] [blame] | 127 | instance_id, 'SUSPENDED') |
Jay Pipes | 051075a | 2012-04-28 17:39:37 -0400 | [diff] [blame] | 128 | |
ivan-zhu | 1997739 | 2013-01-12 21:57:55 +0800 | [diff] [blame] | 129 | def resume_server(self): |
Jay Pipes | 051075a | 2012-04-28 17:39:37 -0400 | [diff] [blame] | 130 | instance = self.get_resource('instance') |
| 131 | instance_id = instance.id |
| 132 | LOG.debug("Resuming instance %s. Current status: %s", |
| 133 | instance_id, instance.status) |
| 134 | instance.resume() |
Sean Dague | 35a7caf | 2013-05-10 10:38:22 -0400 | [diff] [blame] | 135 | self.status_timeout( |
| 136 | self.compute_client.servers, instance_id, 'ACTIVE') |
Jay Pipes | 051075a | 2012-04-28 17:39:37 -0400 | [diff] [blame] | 137 | |
ivan-zhu | 1997739 | 2013-01-12 21:57:55 +0800 | [diff] [blame] | 138 | def terminate_instance(self): |
Jay Pipes | 051075a | 2012-04-28 17:39:37 -0400 | [diff] [blame] | 139 | instance = self.get_resource('instance') |
| 140 | instance.delete() |
| 141 | self.remove_resource('instance') |
ivan-zhu | 1997739 | 2013-01-12 21:57:55 +0800 | [diff] [blame] | 142 | |
Andrea Frittoli | f5da28b | 2013-12-06 07:08:07 +0000 | [diff] [blame] | 143 | def verify_ssh(self): |
| 144 | if self.run_ssh: |
| 145 | # Obtain a floating IP |
| 146 | floating_ip = self.compute_client.floating_ips.create() |
| 147 | # Attach a floating IP |
| 148 | instance = self.get_resource('instance') |
| 149 | instance.add_floating_ip(floating_ip) |
| 150 | # Check ssh |
Nachi Ueno | 95b4128 | 2014-01-15 06:54:21 -0800 | [diff] [blame] | 151 | try: |
| 152 | self.get_remote_client( |
| 153 | server_or_ip=floating_ip.ip, |
| 154 | username=self.image_utils.ssh_user(self.image_ref), |
| 155 | private_key=self.keypair.private) |
| 156 | except Exception: |
| 157 | LOG.exception('ssh to server failed') |
| 158 | self._log_console_output() |
| 159 | raise |
Andrea Frittoli | f5da28b | 2013-12-06 07:08:07 +0000 | [diff] [blame] | 160 | |
Masayuki Igawa | 4ded9f0 | 2014-02-17 15:05:59 +0900 | [diff] [blame] | 161 | @test.services('compute', 'network') |
ivan-zhu | 1997739 | 2013-01-12 21:57:55 +0800 | [diff] [blame] | 162 | def test_server_basicops(self): |
Ken'ichi Ohmichi | 599d1b8 | 2013-08-19 18:48:37 +0900 | [diff] [blame] | 163 | self.add_keypair() |
ivan-zhu | 1997739 | 2013-01-12 21:57:55 +0800 | [diff] [blame] | 164 | self.create_security_group() |
| 165 | self.boot_instance() |
ivan-zhu | 1997739 | 2013-01-12 21:57:55 +0800 | [diff] [blame] | 166 | self.pause_server() |
| 167 | self.unpause_server() |
| 168 | self.suspend_server() |
| 169 | self.resume_server() |
Andrea Frittoli | f5da28b | 2013-12-06 07:08:07 +0000 | [diff] [blame] | 170 | self.verify_ssh() |
ivan-zhu | 1997739 | 2013-01-12 21:57:55 +0800 | [diff] [blame] | 171 | self.terminate_instance() |