blob: e62a52b396ae3695d691bf847248402cb7c3911c [file] [log] [blame]
ZhiQiang Fan39f97222013-09-20 04:49:44 +08001# Copyright 2012 OpenStack Foundation
Jay Pipes13b479b2012-06-11 14:52:27 -04002# 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
Daryl Walleck6b9b2882012-04-08 21:43:39 -050016import base64
Jay Pipes13b479b2012-06-11 14:52:27 -040017
Mauro S. M. Rodrigues54110362013-02-15 13:14:04 -050018import netaddr
ivan-zhu1feeb382013-01-24 10:14:39 +080019import testtools
Jay Pipes13b479b2012-06-11 14:52:27 -040020
Sean Dague1937d092013-05-17 16:36:38 -040021from tempest.api.compute import base
Fei Long Wangd39431f2015-05-14 11:30:48 +120022from tempest.common.utils import data_utils
Masayuki Igawa209fd502014-02-17 14:46:43 +090023from tempest.common.utils.linux import remote_client
Sean Dague86bd8422013-12-20 09:56:44 -050024from tempest import config
Masayuki Igawa209fd502014-02-17 14:46:43 +090025from tempest import test
Daryl Walleck6b9b2882012-04-08 21:43:39 -050026
Sean Dague86bd8422013-12-20 09:56:44 -050027CONF = config.CONF
28
Daryl Walleck6b9b2882012-04-08 21:43:39 -050029
ivan-zhuf2b00502013-10-18 10:06:52 +080030class ServersTestJSON(base.BaseV2ComputeTest):
ivan-zhua5141d92013-03-06 23:12:43 +080031 disk_config = 'AUTO'
Daryl Walleck6b9b2882012-04-08 21:43:39 -050032
Attila Fazekas19044d52013-02-16 07:35:06 +010033 @classmethod
Rohan Kanade60b73092015-02-04 17:58:19 +053034 def setup_credentials(cls):
Attila Fazekas423834d2014-03-14 17:33:13 +010035 cls.prepare_instance_network()
Rohan Kanade60b73092015-02-04 17:58:19 +053036 super(ServersTestJSON, cls).setup_credentials()
37
38 @classmethod
39 def setup_clients(cls):
40 super(ServersTestJSON, cls).setup_clients()
41 cls.client = cls.servers_client
42 cls.network_client = cls.os.network_client
43
44 @classmethod
45 def resource_setup(cls):
nithya-ganesan222efd72015-01-22 12:20:27 +000046 cls.set_validation_resources()
Andrea Frittoli50bb80d2014-09-15 12:34:27 +010047 super(ServersTestJSON, cls).resource_setup()
Daryl Walleck6b9b2882012-04-08 21:43:39 -050048 cls.meta = {'hello': 'world'}
49 cls.accessIPv4 = '1.1.1.1'
Mauro S. M. Rodrigues54110362013-02-15 13:14:04 -050050 cls.accessIPv6 = '0000:0000:0000:0000:0000:babe:220.12.22.2'
Masayuki Igawa259c1132013-10-31 17:48:44 +090051 cls.name = data_utils.rand_name('server')
Daryl Walleck6b9b2882012-04-08 21:43:39 -050052 file_contents = 'This is a test file.'
Pádraig Bradyc6081cf2013-01-04 17:43:53 +000053 personality = [{'path': '/test.txt',
Daryl Walleck6b9b2882012-04-08 21:43:39 -050054 'contents': base64.b64encode(file_contents)}]
David Kranz0fb14292015-02-11 15:55:20 -050055 disk_config = cls.disk_config
Joseph Lanouxb3e1f872015-01-30 11:13:07 +000056 cls.server_initial = cls.create_test_server(
57 validatable=True,
58 wait_until='ACTIVE',
59 name=cls.name,
60 meta=cls.meta,
61 accessIPv4=cls.accessIPv4,
62 accessIPv6=cls.accessIPv6,
63 personality=personality,
64 disk_config=disk_config)
Daryl Walleck6b9b2882012-04-08 21:43:39 -050065 cls.password = cls.server_initial['adminPass']
Ken'ichi Ohmichi76800242015-07-03 05:12:31 +000066 cls.server = cls.client.show_server(cls.server_initial['id'])
Daryl Walleck6b9b2882012-04-08 21:43:39 -050067
Matt Riedemann778b5f92015-03-11 12:44:28 -070068 def _create_net_subnet_ret_net_from_cidr(self, cidr):
69 name_net = data_utils.rand_name(self.__class__.__name__)
70 net = self.network_client.create_network(name=name_net)
71 self.addCleanup(self.network_client.delete_network,
72 net['network']['id'])
73
74 subnet = self.network_client.create_subnet(
75 network_id=net['network']['id'],
76 cidr=cidr,
77 ip_version=4)
78 self.addCleanup(self.network_client.delete_subnet,
79 subnet['subnet']['id'])
80 return net
81
Masayuki Igawa209fd502014-02-17 14:46:43 +090082 @test.attr(type='smoke')
Chris Hoge7579c1a2015-02-26 14:12:15 -080083 @test.idempotent_id('5de47127-9977-400a-936f-abcfbec1218f')
Daryl Wallecked97dca2012-07-04 23:25:45 -050084 def test_verify_server_details(self):
Sean Dague4dd2c0b2013-01-03 17:50:28 -050085 # Verify the specified server attributes are set correctly
Daryl Walleck6b9b2882012-04-08 21:43:39 -050086 self.assertEqual(self.accessIPv4, self.server['accessIPv4'])
Mauro S. M. Rodrigues54110362013-02-15 13:14:04 -050087 # NOTE(maurosr): See http://tools.ietf.org/html/rfc5952 (section 4)
88 # Here we compare directly with the canonicalized format.
89 self.assertEqual(self.server['accessIPv6'],
90 str(netaddr.IPAddress(self.accessIPv6)))
Daryl Walleck6b9b2882012-04-08 21:43:39 -050091 self.assertEqual(self.name, self.server['name'])
92 self.assertEqual(self.image_ref, self.server['image']['id'])
Ken'ichi Ohmichi35772602013-11-14 15:03:27 +090093 self.assertEqual(self.flavor_ref, self.server['flavor']['id'])
Daryl Walleck6b9b2882012-04-08 21:43:39 -050094 self.assertEqual(self.meta, self.server['metadata'])
95
Masayuki Igawa209fd502014-02-17 14:46:43 +090096 @test.attr(type='smoke')
Chris Hoge7579c1a2015-02-26 14:12:15 -080097 @test.idempotent_id('9a438d88-10c6-4bcd-8b5b-5b6e25e1346f')
Daryl Wallecked97dca2012-07-04 23:25:45 -050098 def test_list_servers(self):
Sean Dague4dd2c0b2013-01-03 17:50:28 -050099 # The created server should be in the list of all servers
David Kranzae99b9a2015-02-16 13:37:01 -0500100 body = self.client.list_servers()
Daryl Wallecked97dca2012-07-04 23:25:45 -0500101 servers = body['servers']
102 found = any([i for i in servers if i['id'] == self.server['id']])
103 self.assertTrue(found)
104
Chris Hoge7579c1a2015-02-26 14:12:15 -0800105 @test.idempotent_id('585e934c-448e-43c4-acbf-d06a9b899997')
Daryl Wallecked97dca2012-07-04 23:25:45 -0500106 def test_list_servers_with_detail(self):
Sean Dague4dd2c0b2013-01-03 17:50:28 -0500107 # The created server should be in the detailed list of all servers
Ken'ichi Ohmichicbc26a82015-07-03 08:18:04 +0000108 body = self.client.list_servers(detail=True)
Daryl Wallecked97dca2012-07-04 23:25:45 -0500109 servers = body['servers']
110 found = any([i for i in servers if i['id'] == self.server['id']])
111 self.assertTrue(found)
112
Chris Hoge7579c1a2015-02-26 14:12:15 -0800113 @test.idempotent_id('cbc0f52f-05aa-492b-bdc1-84b575ca294b')
Matthew Treinishe5cca002015-05-11 15:36:50 -0400114 @testtools.skipUnless(CONF.validation.run_validation,
Matt Riedemann6c668202014-03-24 09:17:10 -0700115 'Instance validation tests are disabled.')
Daryl Walleck6b9b2882012-04-08 21:43:39 -0500116 def test_verify_created_server_vcpus(self):
Sean Dague4dd2c0b2013-01-03 17:50:28 -0500117 # Verify that the number of vcpus reported by the instance matches
118 # the amount stated by the flavor
Ken'ichi Ohmichi5628f3f2015-05-22 20:17:56 +0000119 flavor = self.flavors_client.show_flavor(self.flavor_ref)
Joseph Lanouxffe09dd2015-03-18 16:45:33 +0000120 linux_client = remote_client.RemoteClient(
121 self.get_server_ip(self.server),
122 self.ssh_user,
123 self.password,
124 self.validation_resources['keypair']['private_key'])
Daryl Walleck6b9b2882012-04-08 21:43:39 -0500125 self.assertEqual(flavor['vcpus'], linux_client.get_number_of_vcpus())
126
Chris Hoge7579c1a2015-02-26 14:12:15 -0800127 @test.idempotent_id('ac1ad47f-984b-4441-9274-c9079b7a0666')
Matthew Treinishe5cca002015-05-11 15:36:50 -0400128 @testtools.skipUnless(CONF.validation.run_validation,
Matt Riedemann6c668202014-03-24 09:17:10 -0700129 'Instance validation tests are disabled.')
Daryl Walleck6b9b2882012-04-08 21:43:39 -0500130 def test_host_name_is_same_as_server_name(self):
Sean Dague4dd2c0b2013-01-03 17:50:28 -0500131 # Verify the instance host name is the same as the server name
Joseph Lanouxffe09dd2015-03-18 16:45:33 +0000132 linux_client = remote_client.RemoteClient(
133 self.get_server_ip(self.server),
134 self.ssh_user,
135 self.password,
136 self.validation_resources['keypair']['private_key'])
Daryl Walleck6b9b2882012-04-08 21:43:39 -0500137 self.assertTrue(linux_client.hostname_equals_servername(self.name))
Dan Smith4307f992012-08-16 09:23:20 -0700138
Chris Hoge7579c1a2015-02-26 14:12:15 -0800139 @test.idempotent_id('ed20d3fb-9d1f-4329-b160-543fbd5d9811')
Ken'ichi Ohmichi60a67f72014-05-20 09:19:50 +0900140 def test_create_server_with_scheduler_hint_group(self):
141 # Create a server with the scheduler hint "group".
142 name = data_utils.rand_name('server_group')
143 policies = ['affinity']
Ken'ichi Ohmichi7ca54b82015-07-07 01:10:26 +0000144 body = self.server_groups_client.create_server_group(
145 name=name, policies=policies)
Ken'ichi Ohmichi60a67f72014-05-20 09:19:50 +0900146 group_id = body['id']
Ken'ichi Ohmichi7ca54b82015-07-07 01:10:26 +0000147 self.addCleanup(self.server_groups_client.delete_server_group,
148 group_id)
Ken'ichi Ohmichi60a67f72014-05-20 09:19:50 +0900149
150 hints = {'group': group_id}
David Kranz0fb14292015-02-11 15:55:20 -0500151 server = self.create_test_server(sched_hints=hints,
152 wait_until='ACTIVE')
Ken'ichi Ohmichi60a67f72014-05-20 09:19:50 +0900153
154 # Check a server is in the group
Ken'ichi Ohmichi7ca54b82015-07-07 01:10:26 +0000155 server_group = self.server_groups_client.get_server_group(group_id)
Ken'ichi Ohmichi60a67f72014-05-20 09:19:50 +0900156 self.assertIn(server['id'], server_group['members'])
157
Chris Hoge7579c1a2015-02-26 14:12:15 -0800158 @test.idempotent_id('0578d144-ed74-43f8-8e57-ab10dbf9b3c2')
Joseph Lanoux38edc052014-09-18 14:25:46 +0000159 @testtools.skipUnless(CONF.service_available.neutron,
160 'Neutron service must be available.')
161 def test_verify_multiple_nics_order(self):
162 # Verify that the networks order given at the server creation is
163 # preserved within the server.
Matt Riedemann778b5f92015-03-11 12:44:28 -0700164 net1 = self._create_net_subnet_ret_net_from_cidr('19.80.0.0/24')
165 net2 = self._create_net_subnet_ret_net_from_cidr('19.86.0.0/24')
Joseph Lanoux38edc052014-09-18 14:25:46 +0000166
167 networks = [{'uuid': net1['network']['id']},
168 {'uuid': net2['network']['id']}]
169
David Kranz0fb14292015-02-11 15:55:20 -0500170 server_multi_nics = self.create_test_server(
Joseph Lanoux38edc052014-09-18 14:25:46 +0000171 networks=networks, wait_until='ACTIVE')
172
Joseph Lanouxe59f0ff2014-10-17 14:56:56 +0000173 # Cleanup server; this is needed in the test case because with the LIFO
174 # nature of the cleanups, if we don't delete the server first, the port
175 # will still be part of the subnet and we'll get a 409 from Neutron
176 # when trying to delete the subnet. The tear down in the base class
177 # will try to delete the server and get a 404 but it's ignored so
178 # we're OK.
179 def cleanup_server():
180 self.client.delete_server(server_multi_nics['id'])
181 self.client.wait_for_server_termination(server_multi_nics['id'])
182
183 self.addCleanup(cleanup_server)
184
David Kranzae99b9a2015-02-16 13:37:01 -0500185 addresses = self.client.list_addresses(server_multi_nics['id'])
Joseph Lanoux38edc052014-09-18 14:25:46 +0000186
venkata anilbd2d49a2015-01-16 08:07:15 +0000187 # We can't predict the ip addresses assigned to the server on networks.
188 # Sometimes the assigned addresses are ['19.80.0.2', '19.86.0.2'], at
189 # other times ['19.80.0.3', '19.86.0.3']. So we check if the first
190 # address is in first network, similarly second address is in second
191 # network.
Matt Riedemann778b5f92015-03-11 12:44:28 -0700192 addr = [addresses[net1['network']['name']][0]['addr'],
193 addresses[net2['network']['name']][0]['addr']]
venkata anilbd2d49a2015-01-16 08:07:15 +0000194 networks = [netaddr.IPNetwork('19.80.0.0/24'),
195 netaddr.IPNetwork('19.86.0.0/24')]
196 for address, network in zip(addr, networks):
197 self.assertIn(address, network)
Joseph Lanoux38edc052014-09-18 14:25:46 +0000198
Matt Riedemann778b5f92015-03-11 12:44:28 -0700199 @test.idempotent_id('1678d144-ed74-43f8-8e57-ab10dbf9b3c2')
200 @testtools.skipUnless(CONF.service_available.neutron,
201 'Neutron service must be available.')
202 # The below skipUnless should be removed once Kilo-eol happens.
203 @testtools.skipUnless(CONF.compute_feature_enabled.
204 allow_duplicate_networks,
205 'Duplicate networks must be allowed')
206 def test_verify_duplicate_network_nics(self):
207 # Verify that server creation does not fail when more than one nic
208 # is created on the same network.
209 net1 = self._create_net_subnet_ret_net_from_cidr('19.80.0.0/24')
210 net2 = self._create_net_subnet_ret_net_from_cidr('19.86.0.0/24')
211
212 networks = [{'uuid': net1['network']['id']},
213 {'uuid': net2['network']['id']},
214 {'uuid': net1['network']['id']}]
215
216 server_multi_nics = self.create_test_server(
217 networks=networks, wait_until='ACTIVE')
218
219 def cleanup_server():
220 self.client.delete_server(server_multi_nics['id'])
221 self.client.wait_for_server_termination(server_multi_nics['id'])
222
223 self.addCleanup(cleanup_server)
224
225 addresses = self.client.list_addresses(server_multi_nics['id'])
226
227 addr = [addresses[net1['network']['name']][0]['addr'],
228 addresses[net2['network']['name']][0]['addr'],
229 addresses[net1['network']['name']][1]['addr']]
230 networks = [netaddr.IPNetwork('19.80.0.0/24'),
231 netaddr.IPNetwork('19.86.0.0/24'),
232 netaddr.IPNetwork('19.80.0.0/24')]
233 for address, network in zip(addr, networks):
234 self.assertIn(address, network)
235
Dan Smith4307f992012-08-16 09:23:20 -0700236
Chang Bo Guoedd6ec02013-12-12 23:53:09 -0800237class ServersWithSpecificFlavorTestJSON(base.BaseV2ComputeAdminTest):
Chang Bo Guoedd6ec02013-12-12 23:53:09 -0800238 disk_config = 'AUTO'
239
240 @classmethod
Joseph Lanouxffe09dd2015-03-18 16:45:33 +0000241 def setup_credentials(cls):
Attila Fazekas423834d2014-03-14 17:33:13 +0100242 cls.prepare_instance_network()
Joseph Lanouxffe09dd2015-03-18 16:45:33 +0000243 super(ServersWithSpecificFlavorTestJSON, cls).setup_credentials()
244
245 @classmethod
246 def setup_clients(cls):
Rohan Kanade60b73092015-02-04 17:58:19 +0530247 super(ServersWithSpecificFlavorTestJSON, cls).setup_clients()
Chang Bo Guoedd6ec02013-12-12 23:53:09 -0800248 cls.flavor_client = cls.os_adm.flavors_client
Attila Fazekas076f79f2014-03-14 17:43:46 +0100249 cls.client = cls.servers_client
Chang Bo Guoedd6ec02013-12-12 23:53:09 -0800250
Joseph Lanouxffe09dd2015-03-18 16:45:33 +0000251 @classmethod
252 def resource_setup(cls):
253 cls.set_validation_resources()
254
255 super(ServersWithSpecificFlavorTestJSON, cls).resource_setup()
256
Chris Hoge7579c1a2015-02-26 14:12:15 -0800257 @test.idempotent_id('b3c7bcfc-bb5b-4e22-b517-c7f686b802ca')
Matthew Treinishe5cca002015-05-11 15:36:50 -0400258 @testtools.skipUnless(CONF.validation.run_validation,
Matt Riedemann6c668202014-03-24 09:17:10 -0700259 'Instance validation tests are disabled.')
Chang Bo Guoedd6ec02013-12-12 23:53:09 -0800260 def test_verify_created_server_ephemeral_disk(self):
261 # Verify that the ephemeral disk is created when creating server
262
Attila Fazekas076f79f2014-03-14 17:43:46 +0100263 def create_flavor_with_extra_specs():
Chang Bo Guoedd6ec02013-12-12 23:53:09 -0800264 flavor_with_eph_disk_name = data_utils.rand_name('eph_flavor')
265 flavor_with_eph_disk_id = data_utils.rand_int_id(start=1000)
266 ram = 64
267 vcpus = 1
268 disk = 0
269
270 # Create a flavor with extra specs
David Kranz2fa77b22015-02-09 11:39:50 -0500271 flavor = (self.flavor_client.
Ken'ichi Ohmichi96338c42015-07-17 06:25:14 +0000272 create_flavor(name=flavor_with_eph_disk_name,
273 ram=ram, vcpus=vcpus, disk=disk,
274 id=flavor_with_eph_disk_id,
David Kranz2fa77b22015-02-09 11:39:50 -0500275 ephemeral=1))
Attila Fazekas076f79f2014-03-14 17:43:46 +0100276 self.addCleanup(flavor_clean_up, flavor['id'])
Chang Bo Guoedd6ec02013-12-12 23:53:09 -0800277
278 return flavor['id']
279
Attila Fazekas076f79f2014-03-14 17:43:46 +0100280 def create_flavor_without_extra_specs():
Chang Bo Guoedd6ec02013-12-12 23:53:09 -0800281 flavor_no_eph_disk_name = data_utils.rand_name('no_eph_flavor')
282 flavor_no_eph_disk_id = data_utils.rand_int_id(start=1000)
283
284 ram = 64
285 vcpus = 1
286 disk = 0
287
288 # Create a flavor without extra specs
David Kranz2fa77b22015-02-09 11:39:50 -0500289 flavor = (self.flavor_client.
Ken'ichi Ohmichi96338c42015-07-17 06:25:14 +0000290 create_flavor(name=flavor_no_eph_disk_name,
291 ram=ram, vcpus=vcpus, disk=disk,
292 id=flavor_no_eph_disk_id))
Attila Fazekas076f79f2014-03-14 17:43:46 +0100293 self.addCleanup(flavor_clean_up, flavor['id'])
Chang Bo Guoedd6ec02013-12-12 23:53:09 -0800294
295 return flavor['id']
296
Attila Fazekas076f79f2014-03-14 17:43:46 +0100297 def flavor_clean_up(flavor_id):
David Kranz2fa77b22015-02-09 11:39:50 -0500298 self.flavor_client.delete_flavor(flavor_id)
Chang Bo Guoedd6ec02013-12-12 23:53:09 -0800299 self.flavor_client.wait_for_resource_deletion(flavor_id)
300
Attila Fazekas076f79f2014-03-14 17:43:46 +0100301 flavor_with_eph_disk_id = create_flavor_with_extra_specs()
302 flavor_no_eph_disk_id = create_flavor_without_extra_specs()
Chang Bo Guoedd6ec02013-12-12 23:53:09 -0800303
304 admin_pass = self.image_ssh_password
305
David Kranz0fb14292015-02-11 15:55:20 -0500306 server_no_eph_disk = (self.create_test_server(
Joseph Lanouxffe09dd2015-03-18 16:45:33 +0000307 validatable=True,
David Kranz0fb14292015-02-11 15:55:20 -0500308 wait_until='ACTIVE',
309 adminPass=admin_pass,
310 flavor=flavor_no_eph_disk_id))
Joseph Lanouxffe09dd2015-03-18 16:45:33 +0000311
Chang Bo Guoedd6ec02013-12-12 23:53:09 -0800312 # Get partition number of server without extra specs.
Ken'ichi Ohmichi76800242015-07-03 05:12:31 +0000313 server_no_eph_disk = self.client.show_server(
Attila Fazekas076f79f2014-03-14 17:43:46 +0100314 server_no_eph_disk['id'])
Joseph Lanouxffe09dd2015-03-18 16:45:33 +0000315 linux_client = remote_client.RemoteClient(
316 self.get_server_ip(server_no_eph_disk),
317 self.ssh_user,
318 admin_pass,
319 self.validation_resources['keypair']['private_key'])
Attila Fazekas076f79f2014-03-14 17:43:46 +0100320 partition_num = len(linux_client.get_partitions().split('\n'))
Chang Bo Guoedd6ec02013-12-12 23:53:09 -0800321
Joseph Lanouxffe09dd2015-03-18 16:45:33 +0000322 # Explicit server deletion necessary for Juno compatibility
323 self.client.delete_server(server_no_eph_disk['id'])
324
325 server_with_eph_disk = (self.create_test_server(
326 validatable=True,
327 wait_until='ACTIVE',
328 adminPass=admin_pass,
329 flavor=flavor_with_eph_disk_id))
330
Ken'ichi Ohmichi76800242015-07-03 05:12:31 +0000331 server_with_eph_disk = self.client.show_server(
Attila Fazekas076f79f2014-03-14 17:43:46 +0100332 server_with_eph_disk['id'])
Joseph Lanouxffe09dd2015-03-18 16:45:33 +0000333 linux_client = remote_client.RemoteClient(
334 self.get_server_ip(server_with_eph_disk),
335 self.ssh_user,
336 admin_pass,
337 self.validation_resources['keypair']['private_key'])
Attila Fazekas076f79f2014-03-14 17:43:46 +0100338 partition_num_emph = len(linux_client.get_partitions().split('\n'))
339 self.assertEqual(partition_num + 1, partition_num_emph)
Chang Bo Guoedd6ec02013-12-12 23:53:09 -0800340
341
Attila Fazekas19044d52013-02-16 07:35:06 +0100342class ServersTestManualDisk(ServersTestJSON):
343 disk_config = 'MANUAL'
344
Daryl Walleck0aea0032012-12-04 00:53:28 -0600345 @classmethod
Rohan Kanade60b73092015-02-04 17:58:19 +0530346 def skip_checks(cls):
347 super(ServersTestManualDisk, cls).skip_checks()
Sean Dague86bd8422013-12-20 09:56:44 -0500348 if not CONF.compute_feature_enabled.disk_config:
Daryl Walleck0aea0032012-12-04 00:53:28 -0600349 msg = "DiskConfig extension not enabled."
ivan-zhu1feeb382013-01-24 10:14:39 +0800350 raise cls.skipException(msg)