blob: 3dde19f7ecf5fde16a9f87534048fbf7663a929e [file] [log] [blame]
Jay Pipes13b479b2012-06-11 14:52:27 -04001# vim: tabstop=4 shiftwidth=4 softtabstop=4
2
3# Copyright 2012 OpenStack, LLC
4# All Rights Reserved.
5#
6# Licensed under the Apache License, Version 2.0 (the "License"); you may
7# not use this file except in compliance with the License. You may obtain
8# a copy of the License at
9#
10# http://www.apache.org/licenses/LICENSE-2.0
11#
12# Unless required by applicable law or agreed to in writing, software
13# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
14# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
15# License for the specific language governing permissions and limitations
16# under the License.
17
Daryl Walleck73a9e7a2011-11-15 17:43:31 -060018from nose.plugins.attrib import attr
Daryl Walleck73a9e7a2011-11-15 17:43:31 -060019import unittest2 as unittest
Daryl Walleck73a9e7a2011-11-15 17:43:31 -060020
Jay Pipesf38eaac2012-06-21 13:37:35 -040021from tempest.common.utils.data_utils import rand_name, parse_image_id
Brian Waldon738cd632011-12-12 18:45:09 -050022import tempest.config
Jay Pipes13b479b2012-06-11 14:52:27 -040023from tempest import exceptions
Brian Waldon738cd632011-12-12 18:45:09 -050024from tempest import openstack
Jay Pipes13b479b2012-06-11 14:52:27 -040025from tempest.tests.compute.base import BaseComputeTest
Jay Pipesf38eaac2012-06-21 13:37:35 -040026from tempest.tests import compute
Jay Pipes7f757632011-12-02 15:53:32 -050027
Daryl Walleck73a9e7a2011-11-15 17:43:31 -060028
Daryl Walleckc7251962012-03-12 17:26:54 -050029class ImagesTest(BaseComputeTest):
Daryl Walleck73a9e7a2011-11-15 17:43:31 -060030
31 @classmethod
32 def setUpClass(cls):
Jay Pipesf38eaac2012-06-21 13:37:35 -040033 super(ImagesTest, cls).setUpClass()
Daryl Walleckc7251962012-03-12 17:26:54 -050034 cls.client = cls.images_client
35 cls.servers_client = cls.servers_client
Rohit Karajgiea462ae2012-05-27 21:23:21 -070036
Rohit Karajgiea462ae2012-05-27 21:23:21 -070037 cls.image_ids = []
38
Jay Pipesf38eaac2012-06-21 13:37:35 -040039 if compute.MULTI_USER:
40 if cls.config.compute.allow_tenant_isolation:
41 creds = cls._get_isolated_creds()
42 username, tenant_name, password = creds
43 cls.alt_manager = openstack.Manager(username=username,
44 password=password,
45 tenant_name=tenant_name)
Rohit Karajgiea462ae2012-05-27 21:23:21 -070046 else:
Jay Pipesf38eaac2012-06-21 13:37:35 -040047 # Use the alt_XXX credentials in the config file
48 cls.alt_manager = openstack.AltManager()
49 cls.alt_client = cls.alt_manager.images_client
Rohit Karajgiea462ae2012-05-27 21:23:21 -070050
51 def tearDown(self):
52 """Terminate test instances created after a test is executed"""
53 for server in self.servers:
54 resp, body = self.servers_client.delete_server(server['id'])
55 if resp['status'] == '204':
56 self.servers.remove(server)
57 self.servers_client.wait_for_server_termination(server['id'])
58
59 for image_id in self.image_ids:
60 self.client.delete_image(image_id)
61 self.image_ids.remove(image_id)
Daryl Walleck73a9e7a2011-11-15 17:43:31 -060062
Daryl Wallecked8bef32011-12-05 23:02:08 -060063 @attr(type='smoke')
Jay Pipesf38eaac2012-06-21 13:37:35 -040064 @unittest.skipUnless(compute.CREATE_IMAGE_ENABLED,
Brian Waldon738cd632011-12-12 18:45:09 -050065 'Environment unable to create images.')
Daryl Walleck73a9e7a2011-11-15 17:43:31 -060066 def test_create_delete_image(self):
67 """An image for the provided server should be created"""
68 server_name = rand_name('server')
69 resp, server = self.servers_client.create_server(server_name,
70 self.image_ref,
71 self.flavor_ref)
72 self.servers_client.wait_for_server_status(server['id'], 'ACTIVE')
73
Ravikumar Venkatesan94d81172012-01-09 21:53:14 -080074 # Create a new image
Daryl Walleck73a9e7a2011-11-15 17:43:31 -060075 name = rand_name('image')
Daryl Wallecked8bef32011-12-05 23:02:08 -060076 meta = {'image_type': 'test'}
77 resp, body = self.client.create_image(server['id'], name, meta)
Jay Pipesf38eaac2012-06-21 13:37:35 -040078 image_id = parse_image_id(resp['location'])
Daryl Walleck416af922011-11-22 22:28:33 -060079 self.client.wait_for_image_resp_code(image_id, 200)
Daryl Walleck73a9e7a2011-11-15 17:43:31 -060080 self.client.wait_for_image_status(image_id, 'ACTIVE')
81
Ravikumar Venkatesan94d81172012-01-09 21:53:14 -080082 # Verify the image was created correctly
Daryl Walleck73a9e7a2011-11-15 17:43:31 -060083 resp, image = self.client.get_image(image_id)
84 self.assertEqual(name, image['name'])
Daryl Wallecked8bef32011-12-05 23:02:08 -060085 self.assertEqual('test', image['metadata']['image_type'])
86
Ravikumar Venkatesan94d81172012-01-09 21:53:14 -080087 # Verify minRAM and minDisk values are the same as the original image
Daryl Wallecked8bef32011-12-05 23:02:08 -060088 resp, original_image = self.client.get_image(self.image_ref)
89 self.assertEqual(original_image['minRam'], image['minRam'])
90 self.assertEqual(original_image['minDisk'], image['minDisk'])
Daryl Walleck73a9e7a2011-11-15 17:43:31 -060091
Ravikumar Venkatesan94d81172012-01-09 21:53:14 -080092 # Teardown
Daryl Walleck73a9e7a2011-11-15 17:43:31 -060093 self.client.delete_image(image['id'])
94 self.servers_client.delete_server(server['id'])
Ravikumar Venkatesan94d81172012-01-09 21:53:14 -080095
96 @attr(type='negative')
97 def test_create_image_from_deleted_server(self):
Jay Pipes04b70812012-01-11 10:53:24 -050098 """An image should not be created if the server instance is removed """
Ravikumar Venkatesan94d81172012-01-09 21:53:14 -080099 server_name = rand_name('server')
100 resp, server = self.servers_client.create_server(server_name,
101 self.image_ref,
102 self.flavor_ref)
103 self.servers_client.wait_for_server_status(server['id'], 'ACTIVE')
104
105 # Delete server before trying to create server
106 self.servers_client.delete_server(server['id'])
107
108 try:
109 # Create a new image after server is deleted
110 name = rand_name('image')
111 meta = {'image_type': 'test'}
112 resp, body = self.client.create_image(server['id'], name, meta)
113
114 except:
115 pass
116
117 else:
Jay Pipesf38eaac2012-06-21 13:37:35 -0400118 image_id = parse_image_id(resp['location'])
Daryl Walleckade49f92012-01-16 23:14:26 -0600119 self.client.wait_for_image_resp_code(image_id, 200)
120 self.client.wait_for_image_status(image_id, 'ACTIVE')
Jay Pipes04b70812012-01-11 10:53:24 -0500121 self.client.delete_image(image_id)
Daryl Walleckade49f92012-01-16 23:14:26 -0600122 self.fail("Should not create snapshot from deleted instance!")
rajalakshmi-ganesan32f8db62012-05-18 19:13:40 +0530123
124 @attr(type='negative')
125 def test_create_image_from_invalid_server(self):
126 """An image should not be created with invalid server id"""
127 try:
128 # Create a new image with invalid server id
129 name = rand_name('image')
130 meta = {'image_type': 'test'}
131 resp = {}
132 resp['status'] = None
133 resp, body = self.client.create_image('!@#$%^&*()', name, meta)
134
135 except exceptions.NotFound:
136 pass
137
138 finally:
139 if (resp['status'] != None):
Jay Pipesf38eaac2012-06-21 13:37:35 -0400140 image_id = parse_image_id(resp['location'])
rajalakshmi-ganesan32f8db62012-05-18 19:13:40 +0530141 resp, _ = self.client.delete_image(image_id)
142 self.fail("An image should not be created"
143 " with invalid server id")
144
145 @attr(type='negative')
Jay Pipesf38eaac2012-06-21 13:37:35 -0400146 @unittest.skipUnless(compute.MULTI_USER, 'Second user not configured')
Rohit Karajgiea462ae2012-05-27 21:23:21 -0700147 def test_create_image_for_server_in_another_tenant(self):
148 """Creating image of another tenant's server should be return error"""
149 server = self.create_server()
150
151 snapshot_name = rand_name('test-snap-')
152 self.assertRaises(exceptions.NotFound, self.alt_client.create_image,
153 server['id'], snapshot_name)
154
155 @attr(type='negative')
156 def test_create_image_when_server_is_building(self):
157 """Return error when creating an image of a server that is building"""
158 server_name = rand_name('test-vm-')
159 resp, server = self.servers_client.create_server(server_name,
160 self.image_ref,
161 self.flavor_ref)
162 self.servers.append(server)
163 snapshot_name = rand_name('test-snap-')
164 self.assertRaises(exceptions.Duplicate, self.client.create_image,
165 server['id'], snapshot_name)
166
David Kranz0312b692012-08-21 17:51:17 -0400167 @unittest.skip("Until Bug 1039739 is fixed")
Rohit Karajgiea462ae2012-05-27 21:23:21 -0700168 @attr(type='negative')
169 def test_create_image_when_server_is_rebooting(self):
170 """Return error when creating an image of server that is rebooting"""
171 server = self.create_server()
172 self.servers_client.reboot(server['id'], 'HARD')
173
174 snapshot_name = rand_name('test-snap-')
175 self.assertRaises(exceptions.Duplicate, self.client.create_image,
176 server['id'], snapshot_name)
177
178 @attr(type='negative')
179 def test_create_image_when_server_is_terminating(self):
180 """Return an error when creating image of server that is terminating"""
181 server = self.create_server()
182 self.servers_client.delete_server(server['id'])
183
184 snapshot_name = rand_name('test-snap-')
185 self.assertRaises(exceptions.Duplicate, self.client.create_image,
186 server['id'], snapshot_name)
187
188 @attr(type='negative')
189 def test_create_second_image_when_first_image_is_being_saved(self):
190 """Disallow creating another image when first image is being saved"""
191 server = self.create_server()
192
193 try:
194 # Create first snapshot
195 snapshot_name = rand_name('test-snap-')
196 resp, body = self.client.create_image(server['id'], snapshot_name)
Jay Pipesf38eaac2012-06-21 13:37:35 -0400197 image_id = parse_image_id(resp['location'])
Rohit Karajgiea462ae2012-05-27 21:23:21 -0700198 self.image_ids.append(image_id)
199
200 # Create second snapshot
201 alt_snapshot_name = rand_name('test-snap-')
202 self.client.create_image(server['id'], alt_snapshot_name)
203 except exceptions.Duplicate:
204 pass
205
206 else:
207 self.fail("Should allow creating an image when another image of"
208 "the server is still being saved")
209
210 @attr(type='negative')
211 @unittest.skip("Until Bug 1004564 is fixed")
212 def test_create_image_specify_name_over_256_chars(self):
213 """Return an error if snapshot name over 256 characters is passed"""
214 server = self.create_server()
215
216 try:
217 snapshot_name = rand_name('a' * 260)
218 self.assertRaises(exceptions.BadRequest, self.client.create_image,
219 server['id'], snapshot_name)
220 except:
221 self.fail("Should return 400 Bad Request if image name is over 256"
222 " characters")
223
224 @attr(type='negative')
225 @unittest.skip("Until Bug 1005397 is fixed")
226 def test_create_image_specify_uuid_35_characters_or_less(self):
227 """Return an error if Image ID passed is 35 characters or less"""
228 try:
229 snapshot_name = rand_name('test-snap-')
230 test_uuid = ('a' * 35)
231 self.assertRaises(exceptions.BadRequest, self.client.create_image,
232 test_uuid, snapshot_name)
233 except:
234 self.fail("Should return 400 Bad Request if server uuid is 35"
235 " characters or less")
236
237 @attr(type='negative')
238 @unittest.skip("Until Bug 1005397 is fixed")
239 def test_create_image_specify_uuid_37_characters_or_more(self):
240 """Return an error if Image ID passed is 37 characters or more"""
241 try:
242 snapshot_name = rand_name('test-snap-')
243 test_uuid = ('a' * 37)
244 self.assertRaises(exceptions.BadRequest, self.client.create_image,
245 test_uuid, snapshot_name)
246 except:
247 self.fail("Should return 400 Bad Request if server uuid is 37"
248 " characters or more")
249
250 @attr(type='negative')
251 @unittest.skip("Until Bug 1006725 is fixed")
252 def test_create_image_specify_multibyte_character_image_name(self):
253 """Return an error if the image name has multi-byte characters"""
254 server = self.create_server()
255
256 try:
257 snapshot_name = rand_name('\xef\xbb\xbf')
258 self.assertRaises(exceptions.BadRequest,
259 self.client.create_image, server['id'],
260 snapshot_name)
261 except:
262 self.fail("Should return 400 Bad Request if multi byte characters"
263 " are used for image name")
264
265 @attr(type='negative')
266 @unittest.skip("Until Bug 1005423 is fixed")
267 def test_create_image_specify_invalid_metadata(self):
268 """Return an error when creating image with invalid metadata"""
269 server = self.create_server()
270
271 try:
272 snapshot_name = rand_name('test-snap-')
273 meta = {'': ''}
274 self.assertRaises(exceptions.BadRequest, self.client.create_image,
275 server['id'], snapshot_name, meta)
276
277 except:
278 self.fail("Should raise 400 Bad Request if meta data is invalid")
279
280 @attr(type='negative')
281 @unittest.skip("Until Bug 1005423 is fixed")
282 def test_create_image_specify_metadata_over_limits(self):
283 """Return an error when creating image with meta data over 256 chars"""
284 server = self.create_server()
285
286 try:
287 snapshot_name = rand_name('test-snap-')
288 meta = {'a' * 260: 'b' * 260}
289 self.assertRaises(exceptions.OverLimit, self.client.create_image,
290 server['id'], snapshot_name, meta)
291
292 except:
293 self.fail("Should raise 413 Over Limit if meta data was too long")
294
295 @attr(type='negative')
rajalakshmi-ganesan32f8db62012-05-18 19:13:40 +0530296 def test_delete_image_with_invalid_image_id(self):
297 """An image should not be deleted with invalid image id"""
298 try:
299 # Delete an image with invalid image id
300 resp, _ = self.client.delete_image('!@$%^&*()')
301
302 except exceptions.NotFound:
303 pass
304
305 else:
306 self.fail("DELETE image request should rasie NotFound exception"
307 "when requested with invalid image")
Rohit Karajgiea462ae2012-05-27 21:23:21 -0700308
309 @attr(type='negative')
310 def test_delete_non_existent_image(self):
311 """Return an error while trying to delete a non-existent image"""
312
313 non_existent_image_id = '11a22b9-12a9-5555-cc11-00ab112223fa'
314 self.assertRaises(exceptions.NotFound, self.client.delete_image,
315 non_existent_image_id)
316
317 @attr(type='negative')
Rohit Karajgiea462ae2012-05-27 21:23:21 -0700318 def test_delete_image_blank_id(self):
319 """Return an error while trying to delete an image with blank Id"""
320
321 try:
David Kranz28e35c52012-07-10 10:14:38 -0400322 self.assertRaises(exceptions.NotFound, self.client.delete_image,
Rohit Karajgiea462ae2012-05-27 21:23:21 -0700323 '')
324 except:
David Kranz28e35c52012-07-10 10:14:38 -0400325 self.fail("Did not return HTTP 404 NotFound for blank image id")
Rohit Karajgiea462ae2012-05-27 21:23:21 -0700326
327 @attr(type='negative')
Rohit Karajgiea462ae2012-05-27 21:23:21 -0700328 def test_delete_image_non_hex_string_id(self):
329 """Return an error while trying to delete an image with non hex id"""
330
331 image_id = '11a22b9-120q-5555-cc11-00ab112223gj'
332 try:
David Kranz28e35c52012-07-10 10:14:38 -0400333 self.assertRaises(exceptions.NotFound, self.client.delete_image,
Rohit Karajgiea462ae2012-05-27 21:23:21 -0700334 image_id)
335 except:
David Kranz28e35c52012-07-10 10:14:38 -0400336 self.fail("Did not return HTTP 404 NotFound for non hex image")
Rohit Karajgiea462ae2012-05-27 21:23:21 -0700337
338 @attr(type='negative')
Rohit Karajgiea462ae2012-05-27 21:23:21 -0700339 def test_delete_image_negative_image_id(self):
340 """Return an error while trying to delete an image with negative id"""
341
342 try:
David Kranz28e35c52012-07-10 10:14:38 -0400343 self.assertRaises(exceptions.NotFound, self.client.delete_image,
Rohit Karajgiea462ae2012-05-27 21:23:21 -0700344 -1)
345 except:
David Kranz28e35c52012-07-10 10:14:38 -0400346 self.fail("Did not return HTTP 404 NotFound for negative image "
Rohit Karajgiea462ae2012-05-27 21:23:21 -0700347 "id")
348
349 @attr(type='negative')
Rohit Karajgiea462ae2012-05-27 21:23:21 -0700350 def test_delete_image_id_is_over_35_character_limit(self):
351 """Return an error while trying to delete image with id over limit"""
352
353 try:
David Kranz28e35c52012-07-10 10:14:38 -0400354 self.assertRaises(exceptions.NotFound, self.client.delete_image,
Rohit Karajgiea462ae2012-05-27 21:23:21 -0700355 '11a22b9-120q-5555-cc11-00ab112223gj-3fac')
356 except:
David Kranz28e35c52012-07-10 10:14:38 -0400357 self.fail("Did not return HTTP 404 NotFound for image id that "
Rohit Karajgiea462ae2012-05-27 21:23:21 -0700358 "exceeds 35 character ID length limit")
359
360 @attr(type='negative')
Jay Pipesf38eaac2012-06-21 13:37:35 -0400361 @unittest.skipUnless(compute.MULTI_USER, 'Second user not configured')
Rohit Karajgiea462ae2012-05-27 21:23:21 -0700362 def test_delete_image_of_another_tenant(self):
363 """Return an error while trying to delete another tenant's image"""
364
365 server = self.create_server()
366
367 snapshot_name = rand_name('test-snap-')
368 resp, body = self.client.create_image(server['id'], snapshot_name)
Jay Pipesf38eaac2012-06-21 13:37:35 -0400369 image_id = parse_image_id(resp['location'])
Rohit Karajgiea462ae2012-05-27 21:23:21 -0700370 self.image_ids.append(image_id)
371 self.client.wait_for_image_resp_code(image_id, 200)
372 self.client.wait_for_image_status(image_id, 'ACTIVE')
373
374 # Delete image
375 self.assertRaises(exceptions.NotFound,
376 self.alt_client.delete_image, image_id)
377
378 @attr(type='negative')
379 def test_delete_image_that_is_not_yet_active(self):
380 """Return an error while trying to delete an active that is creating"""
381
382 server = self.create_server()
383
384 snapshot_name = rand_name('test-snap-')
385 resp, body = self.client.create_image(server['id'], snapshot_name)
Jay Pipesf38eaac2012-06-21 13:37:35 -0400386 image_id = parse_image_id(resp['location'])
Rohit Karajgiea462ae2012-05-27 21:23:21 -0700387 self.image_ids.append(image_id)
388
389 # Do not wait, attempt to delete the image, ensure it's successful
390 resp, body = self.client.delete_image(image_id)
391 self.assertEqual('204', resp['status'])
392 self.image_ids.remove(image_id)
393
394 self.assertRaises(exceptions.NotFound, self.client.get_image, image_id)