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 | |
Alexander Gubanov | 509e284 | 2015-06-09 15:29:51 +0300 | [diff] [blame^] | 16 | import json |
| 17 | |
Doug Hellmann | 583ce2c | 2015-03-11 14:55:46 +0000 | [diff] [blame] | 18 | from oslo_log import log as logging |
| 19 | |
Matthew Treinish | 6c07229 | 2014-01-29 19:15:52 +0000 | [diff] [blame] | 20 | from tempest import config |
Yaroslav Lobankov | 117a48f | 2015-08-11 11:40:44 +0300 | [diff] [blame] | 21 | from tempest import exceptions |
Sean Dague | 6dbc6da | 2013-05-08 17:49:46 -0400 | [diff] [blame] | 22 | from tempest.scenario import manager |
Matthew Treinish | d75edef | 2014-04-11 15:57:16 -0400 | [diff] [blame] | 23 | from tempest.scenario import utils as test_utils |
Masayuki Igawa | 4ded9f0 | 2014-02-17 15:05:59 +0900 | [diff] [blame] | 24 | from tempest import test |
Jay Pipes | 051075a | 2012-04-28 17:39:37 -0400 | [diff] [blame] | 25 | |
Matthew Treinish | 6c07229 | 2014-01-29 19:15:52 +0000 | [diff] [blame] | 26 | CONF = config.CONF |
| 27 | |
Jay Pipes | 051075a | 2012-04-28 17:39:37 -0400 | [diff] [blame] | 28 | LOG = logging.getLogger(__name__) |
| 29 | |
Matthew Treinish | a0048cb | 2014-04-08 17:44:42 -0400 | [diff] [blame] | 30 | load_tests = test_utils.load_tests_input_scenario_utils |
Andrea Frittoli | f5da28b | 2013-12-06 07:08:07 +0000 | [diff] [blame] | 31 | |
Jay Pipes | 051075a | 2012-04-28 17:39:37 -0400 | [diff] [blame] | 32 | |
Ghanshyam | 5a305c4 | 2014-08-27 14:24:58 +0900 | [diff] [blame] | 33 | class TestServerBasicOps(manager.ScenarioTest): |
Jay Pipes | 051075a | 2012-04-28 17:39:37 -0400 | [diff] [blame] | 34 | |
| 35 | """ |
| 36 | This smoke test case follows this basic set of operations: |
| 37 | |
| 38 | * Create a keypair for use in launching an instance |
| 39 | * Create a security group to control network access in instance |
| 40 | * Add simple permissive rules to the security group |
| 41 | * Launch an instance |
ghanshyam | 416c94c | 2014-10-02 13:47:25 +0900 | [diff] [blame] | 42 | * Perform ssh to instance |
YAMAMOTO Takashi | 1f62af2 | 2015-06-16 03:29:50 +0900 | [diff] [blame] | 43 | * Verify metadata service |
Alexander Gubanov | 509e284 | 2015-06-09 15:29:51 +0300 | [diff] [blame^] | 44 | * Verify metadata on config_drive |
Jay Pipes | 051075a | 2012-04-28 17:39:37 -0400 | [diff] [blame] | 45 | * Terminate the instance |
| 46 | """ |
| 47 | |
Andrea Frittoli | f5da28b | 2013-12-06 07:08:07 +0000 | [diff] [blame] | 48 | def setUp(self): |
| 49 | super(TestServerBasicOps, self).setUp() |
| 50 | # Setup image and flavor the test instance |
| 51 | # Support both configured and injected values |
| 52 | if not hasattr(self, 'image_ref'): |
Matthew Treinish | 6c07229 | 2014-01-29 19:15:52 +0000 | [diff] [blame] | 53 | self.image_ref = CONF.compute.image_ref |
Andrea Frittoli | f5da28b | 2013-12-06 07:08:07 +0000 | [diff] [blame] | 54 | if not hasattr(self, 'flavor_ref'): |
Matthew Treinish | 6c07229 | 2014-01-29 19:15:52 +0000 | [diff] [blame] | 55 | self.flavor_ref = CONF.compute.flavor_ref |
Matthew Treinish | 96cadf4 | 2015-05-14 19:45:59 -0400 | [diff] [blame] | 56 | self.image_utils = test_utils.ImageUtils(self.manager) |
Andrea Frittoli | f5da28b | 2013-12-06 07:08:07 +0000 | [diff] [blame] | 57 | if not self.image_utils.is_flavor_enough(self.flavor_ref, |
| 58 | self.image_ref): |
| 59 | raise self.skipException( |
| 60 | '{image} does not fit in {flavor}'.format( |
| 61 | image=self.image_ref, flavor=self.flavor_ref |
| 62 | ) |
| 63 | ) |
Matthew Treinish | e5cca00 | 2015-05-11 15:36:50 -0400 | [diff] [blame] | 64 | self.run_ssh = CONF.validation.run_validation and \ |
Andrea Frittoli | f5da28b | 2013-12-06 07:08:07 +0000 | [diff] [blame] | 65 | self.image_utils.is_sshable_image(self.image_ref) |
| 66 | self.ssh_user = self.image_utils.ssh_user(self.image_ref) |
| 67 | LOG.debug('Starting test for i:{image}, f:{flavor}. ' |
| 68 | 'Run ssh: {ssh}, user: {ssh_user}'.format( |
| 69 | image=self.image_ref, flavor=self.flavor_ref, |
| 70 | ssh=self.run_ssh, ssh_user=self.ssh_user)) |
| 71 | |
Ken'ichi Ohmichi | 599d1b8 | 2013-08-19 18:48:37 +0900 | [diff] [blame] | 72 | def add_keypair(self): |
| 73 | self.keypair = self.create_keypair() |
Jay Pipes | 051075a | 2012-04-28 17:39:37 -0400 | [diff] [blame] | 74 | |
ivan-zhu | 1997739 | 2013-01-12 21:57:55 +0800 | [diff] [blame] | 75 | def boot_instance(self): |
Andrea Frittoli | f5da28b | 2013-12-06 07:08:07 +0000 | [diff] [blame] | 76 | # Create server with image and flavor from input scenario |
Ken'ichi Ohmichi | 1b3461e | 2014-12-02 03:41:07 +0000 | [diff] [blame] | 77 | security_groups = [{'name': self.security_group['name']}] |
Alexander Gubanov | 509e284 | 2015-06-09 15:29:51 +0300 | [diff] [blame^] | 78 | self.md = {'meta1': 'data1', 'meta2': 'data2', 'metaN': 'dataN'} |
Jay Pipes | 051075a | 2012-04-28 17:39:37 -0400 | [diff] [blame] | 79 | create_kwargs = { |
Ghanshyam | 5a305c4 | 2014-08-27 14:24:58 +0900 | [diff] [blame] | 80 | 'key_name': self.keypair['name'], |
Alexander Gubanov | 509e284 | 2015-06-09 15:29:51 +0300 | [diff] [blame^] | 81 | 'security_groups': security_groups, |
| 82 | 'config_drive': CONF.compute_feature_enabled.config_drive, |
| 83 | 'metadata': self.md |
Jay Pipes | 051075a | 2012-04-28 17:39:37 -0400 | [diff] [blame] | 84 | } |
Matthew Treinish | b7144eb | 2013-12-13 22:57:35 +0000 | [diff] [blame] | 85 | self.instance = self.create_server(image=self.image_ref, |
| 86 | flavor=self.flavor_ref, |
| 87 | create_kwargs=create_kwargs) |
ivan-zhu | 1997739 | 2013-01-12 21:57:55 +0800 | [diff] [blame] | 88 | |
Andrea Frittoli | f5da28b | 2013-12-06 07:08:07 +0000 | [diff] [blame] | 89 | def verify_ssh(self): |
| 90 | if self.run_ssh: |
| 91 | # Obtain a floating IP |
ghanshyam | 9a3a9a2 | 2015-08-18 17:03:55 +0900 | [diff] [blame] | 92 | self.floating_ip = (self.floating_ips_client.create_floating_ip() |
| 93 | ['floating_ip']) |
Ghanshyam | 5a305c4 | 2014-08-27 14:24:58 +0900 | [diff] [blame] | 94 | self.addCleanup(self.delete_wrapper, |
| 95 | self.floating_ips_client.delete_floating_ip, |
YAMAMOTO Takashi | 1f62af2 | 2015-06-16 03:29:50 +0900 | [diff] [blame] | 96 | self.floating_ip['id']) |
Andrea Frittoli | f5da28b | 2013-12-06 07:08:07 +0000 | [diff] [blame] | 97 | # Attach a floating IP |
Ghanshyam | 5a305c4 | 2014-08-27 14:24:58 +0900 | [diff] [blame] | 98 | self.floating_ips_client.associate_floating_ip_to_server( |
YAMAMOTO Takashi | 1f62af2 | 2015-06-16 03:29:50 +0900 | [diff] [blame] | 99 | self.floating_ip['ip'], self.instance['id']) |
Andrea Frittoli | f5da28b | 2013-12-06 07:08:07 +0000 | [diff] [blame] | 100 | # Check ssh |
YAMAMOTO Takashi | 1f62af2 | 2015-06-16 03:29:50 +0900 | [diff] [blame] | 101 | self.ssh_client = self.get_remote_client( |
| 102 | server_or_ip=self.floating_ip['ip'], |
JordanP | 3fe2dc3 | 2014-11-17 13:06:01 +0100 | [diff] [blame] | 103 | username=self.image_utils.ssh_user(self.image_ref), |
| 104 | private_key=self.keypair['private_key']) |
Andrea Frittoli | f5da28b | 2013-12-06 07:08:07 +0000 | [diff] [blame] | 105 | |
YAMAMOTO Takashi | 1f62af2 | 2015-06-16 03:29:50 +0900 | [diff] [blame] | 106 | def verify_metadata(self): |
| 107 | if self.run_ssh and CONF.compute_feature_enabled.metadata_service: |
| 108 | # Verify metadata service |
Yaroslav Lobankov | 117a48f | 2015-08-11 11:40:44 +0300 | [diff] [blame] | 109 | md_url = 'http://169.254.169.254/latest/meta-data/public-ipv4' |
| 110 | |
| 111 | def exec_cmd_and_verify_output(): |
| 112 | cmd = 'curl ' + md_url |
| 113 | floating_ip = self.floating_ip['ip'] |
| 114 | result = self.ssh_client.exec_command(cmd) |
| 115 | if result: |
| 116 | msg = ('Failed while verifying metadata on server. Result ' |
| 117 | 'of command "%s" is NOT "%s".' % (cmd, floating_ip)) |
| 118 | self.assertEqual(floating_ip, result, msg) |
| 119 | return 'Verification is successful!' |
| 120 | |
| 121 | if not test.call_until_true(exec_cmd_and_verify_output, |
| 122 | CONF.compute.build_timeout, |
| 123 | CONF.compute.build_interval): |
| 124 | raise exceptions.TimeoutException('Timed out while waiting to ' |
| 125 | 'verify metadata on server. ' |
| 126 | '%s is empty.' % md_url) |
YAMAMOTO Takashi | 1f62af2 | 2015-06-16 03:29:50 +0900 | [diff] [blame] | 127 | |
Alexander Gubanov | 509e284 | 2015-06-09 15:29:51 +0300 | [diff] [blame^] | 128 | def verify_metadata_on_config_drive(self): |
| 129 | if self.run_ssh and CONF.compute_feature_enabled.config_drive: |
| 130 | # Verify metadata on config_drive |
| 131 | cmd_blkid = 'blkid -t LABEL=config-2 -o device' |
| 132 | dev_name = self.ssh_client.exec_command(cmd_blkid) |
| 133 | dev_name = dev_name.rstrip() |
| 134 | self.ssh_client.exec_command('sudo mount %s /mnt' % dev_name) |
| 135 | cmd_md = 'sudo cat /mnt/openstack/latest/meta_data.json' |
| 136 | result = self.ssh_client.exec_command(cmd_md) |
| 137 | self.ssh_client.exec_command('sudo umount /mnt') |
| 138 | result = json.loads(result) |
| 139 | self.assertIn('meta', result) |
| 140 | msg = ('Failed while verifying metadata on config_drive on server.' |
| 141 | ' Result of command "%s" is NOT "%s".' % (cmd_md, self.md)) |
| 142 | self.assertEqual(self.md, result['meta'], msg) |
| 143 | |
Chris Hoge | 7579c1a | 2015-02-26 14:12:15 -0800 | [diff] [blame] | 144 | @test.idempotent_id('7fff3fb3-91d8-4fd0-bd7d-0204f1f180ba') |
Sean Dague | 3c634d1 | 2015-04-27 12:09:19 -0400 | [diff] [blame] | 145 | @test.attr(type='smoke') |
Masayuki Igawa | 4ded9f0 | 2014-02-17 15:05:59 +0900 | [diff] [blame] | 146 | @test.services('compute', 'network') |
ivan-zhu | 1997739 | 2013-01-12 21:57:55 +0800 | [diff] [blame] | 147 | def test_server_basicops(self): |
Ken'ichi Ohmichi | 599d1b8 | 2013-08-19 18:48:37 +0900 | [diff] [blame] | 148 | self.add_keypair() |
Yair Fried | 1fc32a1 | 2014-08-04 09:11:30 +0300 | [diff] [blame] | 149 | self.security_group = self._create_security_group() |
ivan-zhu | 1997739 | 2013-01-12 21:57:55 +0800 | [diff] [blame] | 150 | self.boot_instance() |
Andrea Frittoli | f5da28b | 2013-12-06 07:08:07 +0000 | [diff] [blame] | 151 | self.verify_ssh() |
YAMAMOTO Takashi | 1f62af2 | 2015-06-16 03:29:50 +0900 | [diff] [blame] | 152 | self.verify_metadata() |
Alexander Gubanov | 509e284 | 2015-06-09 15:29:51 +0300 | [diff] [blame^] | 153 | self.verify_metadata_on_config_drive() |
Ghanshyam | 5a305c4 | 2014-08-27 14:24:58 +0900 | [diff] [blame] | 154 | self.servers_client.delete_server(self.instance['id']) |